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

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技術實現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        有數據成員,無成員函數
data        有數據成員,沒成員函數
function    無數據成員,有成員函數
both        有數據成員,有成員函數
*/

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

void test_pointer_to_data_member() {

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

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

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

    
// 但是,如果某類型沒有對應類型的數據成員,就不能用數據成員指針去指向它。
    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() {
    
// 同理,一旦某個類型不是基本類型,就可以定義指向該類型的成員函數的指針。
    
// 并且字面值0可以隱式轉換到該指針。

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

    
// 如果該類型確實有匹配的成員函數,可以使用該成員函數給指針賦值。
    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個函數模板的名字:
        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就行
        
// 當然使用其他類型的數據成員指針 T C::* (T!=void)
        
// 或者成員函數指針進行測試 T (C::*)(paramter_list),也是可以的
        
// 只是int C::* 寫起來比較方便。

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

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


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

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

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

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

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

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

    
// 一切都因為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定義為它的靜態成員函數。

    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,無論是數據成員指針還是成員函數指針。
        
// 也無論類型,簽名如何。

        template
<typename U>
        
static small_t test();
        
// 注意補救函數現在返回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 侯杰 閱讀(2683) 評論(3)  編輯 收藏 引用

只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   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>
            好男人免费精品视频| 欧美成年人视频| 免费日韩av电影| 久久国产视频网站| 国产精品v欧美精品v日本精品动漫| 久久久久久9| 国产精品五月天| 99天天综合性| 日韩视频在线永久播放| 久久综合色一综合色88| 久久精品国产亚洲a| 国产精品麻豆va在线播放| 亚洲精品资源美女情侣酒店| 91久久精品美女高潮| 久久久久久亚洲精品杨幂换脸| 欧美专区日韩专区| 国产麻豆日韩| 亚洲欧美日韩综合一区| 午夜激情亚洲| 国产麻豆91精品| 亚洲欧美日韩一区二区三区在线| 亚洲午夜免费视频| 欧美性感一类影片在线播放| 99热在这里有精品免费| 中文无字幕一区二区三区| 欧美精品国产精品| 亚洲日本电影在线| 一本到高清视频免费精品| 欧美成人久久| 亚洲精品欧美激情| 亚洲一区二区在线观看视频| 欧美日韩中文字幕在线| 亚洲天堂网在线观看| 欧美亚洲一区在线| 国产伊人精品| 麻豆精品一区二区综合av| 亚洲电影免费| 一区二区三区精品久久久| 欧美午夜电影在线| 性欧美暴力猛交69hd| 久久综合国产精品| 亚洲人成网在线播放| 欧美日韩精品二区第二页| 亚洲一区二区三区久久| 久久久另类综合| 91久久久久久| 国产精品久久久亚洲一区| 欧美在线观看一区二区三区| 欧美激情精品久久久久| 一本色道久久加勒比88综合| 国产精品入口夜色视频大尺度| 久久大逼视频| 亚洲精选大片| 久久综合图片| 一区二区三欧美| 国内精品伊人久久久久av一坑| 免费看成人av| 亚洲欧美网站| 91久久精品国产91久久| 欧美一区二区三区免费视频| 亚洲成人在线网| 国产精品卡一卡二卡三| 久久免费国产精品1| 一本色道久久88精品综合| 麻豆精品91| 亚洲淫性视频| 最新国产成人av网站网址麻豆| 国产精品爱久久久久久久| 久久五月天婷婷| 亚洲一区在线看| 亚洲激情综合| 久久女同精品一区二区| 亚洲午夜激情在线| 91久久精品一区二区别| 国产亚洲综合在线| 欧美日韩一区在线| 免费日韩一区二区| 欧美在线视频观看免费网站| 一本大道久久a久久精二百| 久热精品视频在线观看| 午夜精品久久久久久久99樱桃| 亚洲人成人99网站| 在线成人激情视频| 国产日本亚洲高清| 国产精品久久7| 欧美激情精品久久久久久变态| 欧美在线日韩精品| 亚洲欧美久久| 一区二区三区你懂的| 亚洲三级免费电影| 亚洲第一在线| 欧美激情一区二区三区成人| 久久久久久久久一区二区| 亚洲图片欧美日产| av成人免费观看| 亚洲精品一二三| 亚洲国产成人tv| 在线看片一区| 亚洲电影在线看| 亚洲第一区在线观看| 精品成人国产| 一区二区视频免费在线观看 | 亚洲国产视频直播| 国内久久精品| 在线观看中文字幕亚洲| 黄色欧美成人| 在线不卡a资源高清| 在线成人激情黄色| 一区精品在线| 亚洲经典视频在线观看| 亚洲国产欧美一区二区三区久久| 黑人操亚洲美女惩罚| 黄色成人免费观看| 亚洲国产精品999| 亚洲精品黄网在线观看| 99精品视频免费| 亚洲主播在线观看| 欧美伊人影院| 久久综合激情| 亚洲电影第三页| 亚洲毛片av| 亚洲男人影院| 久久久久久久尹人综合网亚洲| 噜噜噜久久亚洲精品国产品小说| 欧美国产激情二区三区| 欧美日韩高清在线一区| 国产精品视频一二三| 国产一区视频在线看| 最新69国产成人精品视频免费| 日韩视频免费观看高清在线视频 | 亚洲乱码国产乱码精品精98午夜| 亚洲精品美女91| 亚洲一区二区三区免费视频| 久久国产66| 欧美fxxxxxx另类| 日韩视频一区二区三区在线播放| 在线亚洲伦理| 久久免费视频网| 欧美吻胸吃奶大尺度电影| 国产日韩欧美自拍| 亚洲另类在线视频| 欧美在线视频网站| 最新成人av网站| 西西人体一区二区| 欧美精品一区二区视频| 国产人妖伪娘一区91| 91久久综合亚洲鲁鲁五月天| 亚洲欧美在线一区二区| 欧美成人黑人xx视频免费观看| 日韩香蕉视频| 久久一区国产| 国产农村妇女毛片精品久久麻豆| 亚洲激情偷拍| 久久精品伊人| 在线亚洲精品| 欧美精品一区二区三区高清aⅴ| 国产精品综合不卡av| 亚洲免费不卡| 免费永久网站黄欧美| 亚洲一二区在线| 欧美精品久久久久a| 伊人色综合久久天天五月婷| 亚洲免费小视频| 亚洲人成人一区二区在线观看| 欧美专区中文字幕| 国产精品久久夜| 亚洲天堂黄色| 亚洲品质自拍| 嫩草国产精品入口| 激情欧美丁香| 久久精品水蜜桃av综合天堂| 一区二区三区四区五区视频 | 久久久999精品免费| 国产精品欧美日韩一区| 日韩亚洲国产精品| 欧美高清在线视频| 久久精品一本| 精品99一区二区| 久久综合伊人77777麻豆| 亚洲欧美国产不卡| 国产精品久久久久久亚洲毛片 | 亚洲一区二区三区在线| 欧美日韩免费观看中文| 亚洲精品久久| 亚洲国产精品久久久久婷婷884| 久久精品一本| 一区精品在线| 欧美jizz19性欧美| 久久综合电影一区| 亚洲电影av| 欧美激情亚洲| 欧美国产一区二区在线观看| 亚洲人成啪啪网站| 亚洲精品久久视频| 欧美日韩亚洲天堂| 午夜精品久久久久久久99黑人| 中文在线一区| 国产欧美日韩三级| 久久天天狠狠| 久久亚洲国产精品日日av夜夜|