青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

posts - 23,  comments - 94,  trackbacks - 0
/************************************************************************/
/* Copyright (c) 2009, Roc King
All rights reserved.

Redistribution and use in source and binary forms,
    with or without modification, are permitted
    provided that the following conditions are met:

1. Redistributions of source code must retain the above copyright notice,
    this list of conditions and the following disclaimer.

2. Redistributions in binary form must reproduce the above copyright notice,
    this list of conditions and the following disclaimer in the documentation
    and other materials provided with the distribution.

3. Neither the name of the Tju nor the names of its contributors
    may be used to endorse or promote products derived from this software
    without specific prior written permission.

THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS
    AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
    INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
    AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
    IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
    INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
    (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
    LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
    HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
    WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
    (INCLUDING NEGLIGENCE OR OTHERWISE)
    ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
    EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.                                                                     
*/
/************************************************************************/


/**
這份代碼詳細介紹了使用SFINAE技術實現(xiàn)is_not_buildin_type的原理。
按順序由上往下閱讀即可。
*/

#include 
<stdio.h>
#include 
<iostream>


/** ------------------------------------------------------------------------ */
#define DEFINITION(prefix)          \
prefix none {};                     \
prefix data { 
int i; };             \
prefix function { 
void f(int ) {} };    \
prefix both { 
int i; double f() { return 0.0; } };

namespace structures { DEFINITION(struct) }
namespace s = structures;

namespace classes { DEFINITION(class) }
namespace c = classes;

namespace unions { DEFINITION(union) }
namespace u = unions;

#undef DEFINITION

/**
上面對class、struct、union分別定義了:
none        有數(shù)據(jù)成員,無成員函數(shù)
data        有數(shù)據(jù)成員,沒成員函數(shù)
function    無數(shù)據(jù)成員,有成員函數(shù)
both        有數(shù)據(jù)成員,有成員函數(shù)
*/

/** ------------------------------------------------------------------------ */

void test_pointer_to_data_member() {

    
// 一旦某個類型不是基本數(shù)據(jù)類型,就可以定義成員指針(數(shù)據(jù)成員指針,成員函數(shù)指針)。
    
// 即使它沒有數(shù)據(jù)成員或者成員函數(shù)。

    
// s::none 并沒有數(shù)據(jù)成員或者成員函數(shù)。
    int s::none::* p; //但是可以定義一個指向s的數(shù)據(jù)成員指針,只要類型不是void
    
// void s::none::* p2; //error C2182: 'p2' : illegal use of type 'void'

    
// 同時,在C++中,字面值0可以隱式轉換到任何指針類型。
    p = 0// ok
    double s::none::* p3 = 0// ok

    
// 但是,如果某類型沒有對應類型的數(shù)據(jù)成員,就不能用數(shù)據(jù)成員指針去指向它。
    int s::data::* p4 = 0;
    p4 
= &s::data::i;  // ok
    double s::data::* p5 = 0;
    
// p5 = &s::data::i;
    
// error C2440: '=' : cannot convert from 'int structures::data::* ' to 'double structures::data::* '

    (
void)p3; (void)p5;
}

// 這個是比較完整的測試
void test_pointer_to_data_member_integrate();


/** ------------------------------------------------------------------------ */

void test_pointer_to_member_function() {
    
// 同理,一旦某個類型不是基本類型,就可以定義指向該類型的成員函數(shù)的指針。
    
// 并且字面值0可以隱式轉換到該指針。

    
int (u::none::* p1)(void)  = 0;
    
double (u::none::* p2)(double= 0;

    
// 如果該類型確實有匹配的成員函數(shù),可以使用該成員函數(shù)給指針賦值。
    double (u::both::* p3 )(void= &u::both::f;
    
void (u::function::* p4)(int= &u::function::f;

    
// 否則不能賦值
    
// double (u::both::* p5 )(void) = &u::function::f;
    
//error C2440: 'initializing' : cannot convert from 'void (__thiscall unions::function::* )(int)' to 'double (__thiscall unions::both::* )(void)'
    
// void (u::function::* p6)(int) = &u::both::f;
    
//error C2440: 'initializing' : cannot convert from 'double (__thiscall unions::both::* )(void)' to 'void (__thiscall unions::function::* )(int)'

    (
void)p1; (void)p2; (void)p3; (void)p4;
}

// 這個是比較完整的測試
void test_pointer_to_member_function_integrate();

/** ------------------------------------------------------------------------ */
/**
那么,測試一個類型是否是內建類型的“一個”方法就是
*/

namespace SFINAE {

    
class true_type { char dummy; true_type(); };
    
class false_type { char dummy[2]; false_type(); };
    
// sizeof(true_type)!=sizeof(false_type)

    template
<class C>
    true_type is_not_buildin_type_test(
int C::* pointer_to_data_member);

    template
<typename T>
    false_type is_not_buildin_type_test();

    
void test_theory() {
        
using namespace std;

        
/* 在當前名字空間下, is_not_buildin_type_test是2個函數(shù)模板的名字:
        template<class C>
        true_type is_not_buildin_type_test(int C::* pointer_to_data_member);
        (以下簡稱模板1)

        template<typename T>
        false_type is_not_buildin_type_test();
        (以下簡稱模板2)

        它們相互構成重載。
        
*/

        cout
<<sizeof( is_not_buildin_type_test<c::none>(0) )<<endl;
        
// 0 可以隱式轉化成 int c::none::* ,可以匹配模板1 (with C = c::none)

        
// 這里int是無關緊要的, 只要不是void就行
        
// 當然使用其他類型的數(shù)據(jù)成員指針 T C::* (T!=void)
        
// 或者成員函數(shù)指針進行測試 T (C::*)(paramter_list),也是可以的
        
// 只是int C::* 寫起來比較方便。

        
// 因為() 可以匹配任何類型的對象,
        
// 所以 0 也可以匹配模板2

        
// 又因為()處于重載選擇優(yōu)先級中的最底層,所以最終匹配模板1。
        
// 注意,此處模板2不能使用(int),或者(T*)因為它的優(yōu)先級高于(int C::*)


        cout
<<sizeof( is_not_buildin_type_test<double>(0) )<<endl;
        
// 0 不能隱式轉換成 int double::*,也就不能匹配模板1 ( with C=double )

        
// 但是還有一個“補救”的函數(shù)模板2,可以匹配任何類型的對象。
        
// 又因為SFINAE(Substitution failure is not an error)機制
        
// 所以對模板1的失敗的匹配并不報錯。
    }

    
// 還有一些細節(jié)
    
// 比如is_not_buildin_type_test并沒有實現(xiàn)。
    
// 但是因為它同時也沒有被調用, 而僅僅是用sizeof測試它的返回值,所以不算錯誤。
    
// 也防止了客戶無意調用這個函數(shù)。

    
// 如何得知哪個is_not_buildin_type_test被重載選中?
    
// 是通過返回值的大小不同來區(qū)分的。

    
// 所以就需要true_type和false_type這2個東西。
    
// 其實更合理的命名應該是small_type和big_type。
    
// 同時,它們聲明有私有的構造函數(shù),并且不實現(xiàn),防止客戶使用這2個類。

    
// 還因為is_not_buildin_type_test并沒有真正實現(xiàn)
    
// 也就沒有真正返回true_type或者false_type
    
// 所以沒有實現(xiàn)true_type和false_type的構造函數(shù)也沒關系。

    
// 一切都因為sizeof ……

    
/** -------------------------------------------------------------------- */
    
/**
    將這種方法再包裝一下
    (避免客戶去使用sizeof( xxx ) == sizeof( ture_type )等等)
    就得到
    
*/

    template
<typename T>
    
class is_not_buildin_type {
        is_not_buildin_type();
    
public:
        
enum { value =
            
sizeof(true_type)==sizeof( is_not_buildin_type_test<T>(0) ) };
    };
    
// 或者將true_type,false_type,定義為它的內嵌類型。
    
// 同時將is_not_buildin_type_test定義為它的靜態(tài)成員函數(shù)。

    template
<typename T>
    
class is_not_buildin_type2 {
        is_not_buildin_type2();

        
// 因為是內嵌的private,客戶不能訪問
        
// 所以可以隨意一點
        typedef char small_t;
        
struct big_t { small_t dummy[2]; };

        template
<typename U>
        
static big_t test(void (U::*)(shortfloat) );
        
// 只要是成員指針就ok,無論是數(shù)據(jù)成員指針還是成員函數(shù)指針。
        
// 也無論類型,簽名如何。

        template
<typename U>
        
static small_t test();
        
// 注意補救函數(shù)現(xiàn)在返回small_t

    
public:
        
// 但這也是無關緊要的,因為small_t和big_t只是告之哪個重載被選中的方式。
        
// 只要這里處理好對應就可以了。
        enum { value= sizeof(big_t)==sizeof( test<T>(0) ) };
    };

    
void test_wrapper() {
        
using namespace std;
        cout
<<is_not_buildin_type<c::data>::value<<endl;
        cout
<<is_not_buildin_type<u::both>::value<<endl;
        cout
<<is_not_buildin_type<float>::value<<endl;

        cout
<<is_not_buildin_type2<c::data>::value<<endl;
        cout
<<is_not_buildin_type2<u::both>::value<<endl;
        cout
<<is_not_buildin_type2<float>::value<<endl;
    }

    
// 一個更完整的測試
    void test_wrapper_integrate();
}

/** ------------------------------------------------------------------------ */
/**測試一個類型是否是內建類型的另一個方法,需要更少的技巧。*/

namespace partial_specialization {

    template
<typename T>
    
struct is_not_buildin_type { enum { value=true }; };
    
// T不是一個內建類型

    
// 除非
    template<>
    
struct is_not_buildin_type<int> { enum { value=false}; };
    
// T是int
    template<>
    
struct is_not_buildin_type<unsigned int> { enum { value=false}; };
    
// T是unsigned int
    
// .. more ..
}

int main()
{
    
using namespace std;
    test_pointer_to_data_member();
    test_pointer_to_data_member_integrate();
    test_pointer_to_member_function();
    test_pointer_to_member_function_integrate();

    cout
<<endl;
    SFINAE::test_theory();
    cout
<<endl;
    SFINAE::test_wrapper();
    cout
<<endl;
    SFINAE::test_wrapper_integrate();
}



void test_pointer_to_data_member_integrate() {
    
// to do
}
void test_pointer_to_member_function_integrate() {
    
// to do
}

namespace SFINAE {
    
void test_wrapper_integrate() {
        
// to do
    }
}

這個代碼已經能很完美的解釋了~
今天看了C++ Templates 的15章.. 收獲頗多.. 以前的很多疑惑都比較開朗了~

posted on 2009-03-16 23:32 Charlie 侯杰 閱讀(2682) 評論(3)  編輯 收藏 引用

只有注冊用戶登錄后才能發(fā)表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


by Charlie
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            欧美日韩aaaaa| 国产精品免费观看视频| 亚洲激情在线观看视频免费| 麻豆精品在线视频| 久久综合色一综合色88| 日韩一区二区精品在线观看| 亚洲精品一区二区三区99| 欧美日韩一卡| 欧美一区二区视频97| 久久国产精品网站| 91久久精品国产91性色| 亚洲国产精品欧美一二99| 免费亚洲一区| 欧美日本亚洲视频| 欧美在线视频导航| 久久综合色一综合色88| 一级日韩一区在线观看| 亚洲欧美另类国产| 亚洲国内自拍| 亚洲综合色噜噜狠狠| 黄网站免费久久| av成人手机在线| 国外成人在线视频| 亚洲最新中文字幕| 悠悠资源网久久精品| 亚洲精品中文字幕在线| 含羞草久久爱69一区| 日韩视频三区| 欧美国产精品v| 欧美亚洲视频| 欧美日本免费一区二区三区| 欧美在线亚洲| 国产精品video| 亚洲视频在线视频| 久久综合精品国产一区二区三区| 在线一区二区三区四区| 久久久久国产一区二区三区四区 | 欧美一区日韩一区| 99www免费人成精品| 欧美伊久线香蕉线新在线| 亚洲网在线观看| 欧美成人69av| 久久永久免费| 国产日韩欧美三级| 亚洲一区免费网站| 在线综合欧美| 欧美国产日韩在线观看| 免费成人黄色片| 国产午夜亚洲精品不卡| 正在播放亚洲一区| 在线亚洲精品福利网址导航| 免费试看一区| 欧美激情性爽国产精品17p| 国产一区二区三区av电影| 久久精品人人做人人综合 | 欧美在线一区二区三区| 亚洲在线视频观看| 欧美高清hd18日本| 亚洲第一精品夜夜躁人人爽 | 国产精品福利久久久| 亚洲国产天堂久久综合网| 在线观看一区视频| 久久久久国产一区二区三区| 久久精品国产99国产精品澳门| 国产精品视频免费观看www| 亚洲特黄一级片| 午夜精品美女久久久久av福利| 欧美日本三级| 一区二区三区欧美激情| 亚洲淫性视频| 国产欧美一区二区三区在线老狼| 一区二区三区偷拍| 亚洲欧美电影在线观看| 久久久久国产一区二区三区四区| 久久精品夜色噜噜亚洲a∨| 国产视频在线观看一区 | 欧美成人亚洲| 亚洲精品久久久久中文字幕欢迎你| 久久久水蜜桃| 亚洲三级影院| 午夜国产精品视频| 国产午夜亚洲精品理论片色戒| 欧美在线三级| 亚洲国产婷婷香蕉久久久久久| 亚洲精品一区在线观看香蕉| 欧美日韩伦理在线| 亚洲女ⅴideoshd黑人| 久久国产精品一区二区三区四区| 国产综合色产在线精品| 久久天堂国产精品| 欧美在线观看网址综合| 国产一区二区三区在线观看免费视频| 久久九九全国免费精品观看| 欧美韩日一区二区三区| 中文精品在线| 国语精品中文字幕| 欧美日本乱大交xxxxx| 亚洲欧美日韩天堂| 欧美国产精品劲爆| 亚洲欧美国产视频| 亚洲第一网站| 国产精品久久久久久久久免费桃花| 欧美一区二区久久久| 亚洲第一级黄色片| 欧美在线亚洲综合一区| 亚洲精品偷拍| 国产视频观看一区| 欧美日韩国产综合新一区| 久久国产精品毛片| 亚洲天堂av电影| 亚洲国产va精品久久久不卡综合| 亚洲欧美日韩久久精品| 亚洲日本视频| 在线日韩av| 国产女优一区| 国产精品国产亚洲精品看不卡15| 久久综合福利| 久久精品理论片| 亚洲欧美日韩国产综合| 亚洲欧洲在线看| 欧美va天堂| 久久精品夜夜夜夜久久| 亚洲欧美电影在线观看| 亚洲作爱视频| 亚洲美女av电影| 亚洲第一福利在线观看| 国产色产综合色产在线视频| 欧美日韩国产在线一区| 欧美69wwwcom| 免费观看在线综合| 久久女同精品一区二区| 久久精品99国产精品| 亚洲女人天堂av| 亚洲一区二区伦理| 9l国产精品久久久久麻豆| 91久久在线播放| 亚洲国产你懂的| 亚洲国产一区在线| 亚洲成人在线网站| 亚洲一区二区三区三| 亚洲另类自拍| 一本色道久久88综合日韩精品| 亚洲欧洲日本专区| 亚洲人成网站色ww在线| 亚洲欧洲一区二区在线观看| 亚洲福利视频一区| 亚洲欧洲综合另类| 亚洲精品中文字幕女同| 亚洲精品在线一区二区| aaa亚洲精品一二三区| 亚洲美女视频在线免费观看| 亚洲精品午夜精品| 国产精品99久久99久久久二8 | 亚洲深夜激情| 亚洲欧美视频在线观看| 午夜精品一区二区三区在线| 欧美一区二区三区电影在线观看| 香蕉av777xxx色综合一区| 久久精品国产亚洲精品| 快射av在线播放一区| 欧美黄色一级视频| 国产精品久久久久9999| 国产一区二区三区网站| 亚洲国产一区二区精品专区| 亚洲美女黄网| 午夜精品婷婷| 欧美成人国产va精品日本一级| 欧美高清在线视频观看不卡| 亚洲国产专区| 亚洲欧美综合国产精品一区| 欧美怡红院视频| 欧美激情视频在线播放 | 久久久久久香蕉网| 欧美了一区在线观看| 国产精品视频xxxx| 亚洲国产91| 亚洲免费一在线| 久久亚洲综合网| 一二美女精品欧洲| 欧美一区二区三区在线免费观看| 久久午夜羞羞影院免费观看| 欧美日韩和欧美的一区二区| 国产婷婷一区二区| 在线视频精品一区| 裸体女人亚洲精品一区| 亚洲精品亚洲人成人网| 欧美一区视频| 国产精品成人一区| 最新日韩中文字幕| 久久免费视频在线观看| 亚洲美女中出| 免费在线播放第一区高清av| 国产欧美日韩在线播放| 亚洲精品视频免费| 久久人人爽国产| 亚洲一区久久| 欧美日韩一视频区二区| 91久久国产自产拍夜夜嗨| 久久久久国色av免费观看性色| 亚洲美女福利视频网站|