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

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.                                                                     
*/
/************************************************************************/


/**
這份代碼詳細(xì)介紹了使用SFINAE技術(shù)實(shí)現(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可以隱式轉(zhuǎn)換到任何指針類型。
    p = 0// ok
    double s::none::* p3 = 0// ok

    
// 但是,如果某類型沒有對應(yīng)類型的數(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可以隱式轉(zhuǎn)換到該指針。

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

    
// 如果該類型確實(shí)有匹配的成員函數(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();

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

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;

        
/* 在當(dāng)前名字空間下, 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)

        它們相互構(gòu)成重載。
        
*/

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

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

        
// 因?yàn)?) 可以匹配任何類型的對象,
        
// 所以 0 也可以匹配模板2

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


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

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

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

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

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

    
// 還因?yàn)閕s_not_buildin_type_test并沒有真正實(shí)現(xiàn)
    
// 也就沒有真正返回true_type或者false_type
    
// 所以沒有實(shí)現(xiàn)true_type和false_type的構(gòu)造函數(shù)也沒關(guān)系。

    
// 一切都因?yàn)閟izeof ……

    
/** -------------------------------------------------------------------- */
    
/**
    將這種方法再包裝一下
    (避免客戶去使用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,定義為它的內(nèi)嵌類型。
    
// 同時將is_not_buildin_type_test定義為它的靜態(tài)成員函數(shù)。

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

        
// 因?yàn)槭莾?nèi)嵌的private,客戶不能訪問
        
// 所以可以隨意一點(diǎn)
        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();
        
// 注意補(bǔ)救函數(shù)現(xiàn)在返回small_t

    
public:
        
// 但這也是無關(guān)緊要的,因?yàn)閟mall_t和big_t只是告之哪個重載被選中的方式。
        
// 只要這里處理好對應(yīng)就可以了。
        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();
}

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

namespace partial_specialization {

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

    
// 除非
    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
    }
}

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

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

只有注冊用戶登錄后才能發(fā)表評論。
網(wǎng)站導(dǎo)航: 博客園   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>
            亚洲一区欧美| 亚洲五月六月| 欧美精品免费在线观看| 免费看的黄色欧美网站| 女同一区二区| 欧美国产亚洲视频| 欧美久色视频| 国产精品videosex极品| 国产精品欧美日韩一区| 国产亚洲综合在线| 曰本成人黄色| 欧美在线二区| 久久只精品国产| 欧美黄网免费在线观看| 亚洲午夜视频在线| 久久米奇亚洲| 欧美视频在线播放| 在线免费观看视频一区| 亚洲一级免费视频| 久久综合久久久久88| 91久久久久久久久| 欧美激情影院| 亚洲中午字幕| 欧美好骚综合网| 国产视频精品免费播放| 国产精品99一区| 亚洲大片av| 性做久久久久久久久| 亚洲国产精品成人综合色在线婷婷 | 欧美不卡视频一区发布| 欧美视频成人| 亚洲国产精品久久久久婷婷老年 | 欧美精品综合| 国产日韩欧美精品综合| 在线亚洲国产精品网站| 欧美成人精品一区| 欧美在线不卡视频| 国产精品资源| 亚洲欧美日韩精品久久亚洲区 | 久久久久国产一区二区| 国产精品乱码一区二三区小蝌蚪| 亚洲国产精品va在线看黑人| 欧美亚洲三区| 欧美激情91| 久久精品噜噜噜成人av农村| 国产精品国产三级国产a| 亚洲电影免费观看高清完整版| 校园春色综合网| 一本色道久久综合亚洲精品不卡| 免费成人黄色av| **网站欧美大片在线观看| 欧美一区视频| 亚洲综合色网站| 国产精品一二三四| 亚洲欧美日韩精品久久| 一本色道久久综合亚洲精品按摩 | 午夜国产不卡在线观看视频| 一本色道久久综合亚洲二区三区| 欧美在线首页| 亚洲一区二区三区四区五区午夜| 欧美日韩视频在线一区二区观看视频| 亚洲黄网站在线观看| 久久婷婷麻豆| 午夜日韩电影| 在线播放一区| 欧美二区视频| 欧美一区影院| 亚洲伊人网站| 一区二区三区视频在线| 国产精品成人一区二区网站软件| 国产精品久久久久一区二区| 国产亚洲电影| 欧美日韩高清在线观看| 欧美黄色小视频| 国产欧美日韩精品在线| 欧美中文字幕视频| 久久综合国产精品| 久久国产精品久久久久久久久久 | 亚洲欧美在线免费| 欧美日韩网站| 一区二区欧美日韩| 亚洲一区二区三区国产| 国产一二精品视频| 亚洲电影激情视频网站| 欧美精品一区三区在线观看| 亚洲尤物影院| 久久久xxx| 一区二区三区高清在线观看| 亚洲伊人第一页| 亚洲精品乱码久久久久久蜜桃麻豆 | 国产精品视频999| 久久久综合精品| 欧美韩国在线| 久久久久久网站| 欧美激情一区二区在线| 欧美一区三区三区高中清蜜桃| 久久一区亚洲| 午夜精品免费视频| 欧美成人69av| 久久国产一区二区三区| 老牛影视一区二区三区| 先锋影音久久久| 免费中文字幕日韩欧美| 久久国产婷婷国产香蕉| 欧美色视频在线| 欧美激情一区二区在线 | 欧美成人午夜激情| 久久狠狠亚洲综合| 欧美久久婷婷综合色| 久久免费黄色| 国产精品视频免费观看| 亚洲区免费影片| 亚洲国产mv| 久久国产综合精品| 欧美一级久久| 国产精品国产精品国产专区不蜜| 欧美激情视频给我| 伊人狠狠色j香婷婷综合| 亚洲欧美成人综合| 亚洲一区二区三| 欧美激情视频一区二区三区在线播放| 久久久久五月天| 国产精品手机视频| 亚洲一级特黄| 欧美午夜不卡在线观看免费 | 亚洲一区久久| 亚洲影视在线| 欧美午夜国产| 亚洲美女黄色| 一区二区三区精品在线| 欧美成人有码| 亚洲精品一区二区三区福利| 亚洲美女av网站| 欧美经典一区二区三区| 亚洲欧洲精品一区二区| 亚洲精品一区在线| 欧美精品手机在线| 亚洲九九爱视频| 在线亚洲精品| 国产精品一级| 欧美在线3区| 噜噜噜躁狠狠躁狠狠精品视频| 国产精品乱码一区二区三区| 亚洲午夜在线| 亚洲欧美另类在线| 国产九区一区在线| 欧美一区二区三区免费看 | 麻豆久久婷婷| 亚洲高清免费在线| 日韩视频专区| 国产精品二区三区四区| 亚洲一区二区在线免费观看视频 | 欧美在线不卡视频| 欧美成人蜜桃| 亚洲激情偷拍| 欧美电影免费观看网站| 99天天综合性| 欧美一区午夜精品| 欧美一区二区视频在线观看2020| 国产精品h在线观看| 亚洲天堂成人在线视频| 国产免费亚洲高清| 麻豆av福利av久久av| 99精品视频免费观看| 欧美午夜久久| 久久蜜桃av一区精品变态类天堂| 亚洲国产三级网| 国产精品一区在线观看你懂的| 欧美一区二区三区的| 亚洲国产精彩中文乱码av在线播放| 一区二区三区av| 黑人巨大精品欧美一区二区| 欧美精品日韩| 欧美在线视频导航| 日韩小视频在线观看专区| 久久久久久久久久久久久女国产乱| 亚洲福利专区| 国产精品久久久久久模特| 久久这里有精品视频| 午夜在线a亚洲v天堂网2018| 亚洲福利在线观看| 久久精品成人欧美大片古装| 日韩视频在线观看国产| 国产自产v一区二区三区c| 欧美日韩国产欧| 久久久99国产精品免费| 亚洲欧美另类在线| 99精品国产高清一区二区| 欧美大尺度在线| 久久久久久亚洲精品不卡4k岛国| 亚洲另类一区二区| 一区在线免费观看| 国产日产亚洲精品| 国产精品a级| 欧美日韩亚洲一区二区三区四区| 免费人成精品欧美精品| 久久精品国产清自在天天线| 亚洲免费影院| 亚洲一区二区高清视频| 久久一区亚洲|