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

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>
            亚洲一区二区高清视频| 欲香欲色天天天综合和网| 国产精自产拍久久久久久| 一区二区三区在线观看欧美| 亚洲成色777777在线观看影院| 亚洲理论电影网| 亚洲韩国日本中文字幕| 性做久久久久久久久| 欧美激情日韩| 一本综合久久| 亚洲欧美日韩在线高清直播| 欧美成年人网| 一区二区三区免费看| 美脚丝袜一区二区三区在线观看 | 国产精品日韩欧美一区二区| 一区精品在线播放| 亚洲国产精品成人久久综合一区| 亚洲一本大道在线| 黄色影院成人| 亚洲精品日韩综合观看成人91| 久久久久久久久岛国免费| 国产精品护士白丝一区av| 亚洲人成网站999久久久综合| 久久精品视频免费| 欧美精品福利在线| 欧美专区第一页| 欧美成人一区二区三区片免费| 黄色成人在线网站| 亚洲六月丁香色婷婷综合久久| 欧美aⅴ一区二区三区视频| 伊人精品视频| 亚洲校园激情| 99re66热这里只有精品4| 亚洲国产欧美一区二区三区久久 | 亚洲一二三区在线| 久久精品人人做人人综合| 国产一区二区三区在线观看网站| 欧美一区二区精美| 免费在线亚洲| 久久天天躁狠狠躁夜夜av| 久久精品国产99| 亚洲一级片在线观看| 99精品视频免费观看视频| 欧美午夜精品理论片a级大开眼界 欧美午夜精品理论片a级按摩 | 一区二区日韩免费看| 欧美日韩久久久久久| 一区二区三区日韩欧美| 一区二区三区欧美亚洲| 国产精品亚洲成人| 日韩午夜三级在线| 亚洲日本免费| 免费91麻豆精品国产自产在线观看| 亚洲国产精品国自产拍av秋霞| 亚洲二区免费| 精品成人国产| 久久久久国色av免费看影院| 亚洲第一在线视频| 欧美一区二视频在线免费观看| ●精品国产综合乱码久久久久 | 国产日本欧美视频| 亚洲视频一区在线观看| 国产一区二区三区av电影| 欧美二区在线看| 黄色一区二区三区| 久久精品亚洲一区| 麻豆成人综合网| 亚洲第一福利视频| 玖玖玖免费嫩草在线影院一区| 一本高清dvd不卡在线观看| 亚洲影院色在线观看免费| 在线观看成人网| 老司机精品福利视频| 亚洲天堂男人| 国产精品欧美经典| 亚洲欧美在线aaa| 久久成人精品| 在线观看日韩av| 欧美成年人视频| 日韩一级在线| 欧美一级午夜免费电影| 欧美大尺度在线观看| 久久激情五月激情| 精品二区视频| 欧美国产日韩一区二区| 久久婷婷久久| 亚洲人成精品久久久久| 久久国产一区二区三区| 亚洲字幕在线观看| 国产午夜亚洲精品羞羞网站| 日韩视频免费观看| 午夜视频久久久| 韩国三级电影久久久久久| 亚洲午夜精品在线| 久久免费视频网站| 99re66热这里只有精品4| 久久人人97超碰国产公开结果| 欧美自拍丝袜亚洲| 亚洲高清二区| 欧美色综合网| 久久躁狠狠躁夜夜爽| 久久亚洲欧美| 在线视频欧美精品| 一区二区三区在线观看国产| 欧美亚洲视频| 亚洲黄网站在线观看| 亚洲三级毛片| 国产精品欧美久久| 麻豆久久精品| 亚洲一区影音先锋| 最新成人在线| 久久久综合网| 午夜亚洲性色视频| 夜夜嗨av一区二区三区四季av| 欧美成年人在线观看| 亚洲国产精品一区在线观看不卡| 一区精品在线播放| 国产精品自拍网站| 欧美精品v国产精品v日韩精品| 亚洲观看高清完整版在线观看| 亚洲欧洲精品一区二区三区不卡| 鲁大师影院一区二区三区| 欧美激情一区二区三区在线视频观看 | 欧美福利网址| 久久国产精品久久国产精品| 国产在线视频欧美| 久久久午夜视频| 午夜精品福利一区二区蜜股av| 久久久精品日韩欧美| 在线看欧美日韩| 国产目拍亚洲精品99久久精品 | 亚洲一级二级在线| 亚洲欧洲日本mm| **网站欧美大片在线观看| 欧美高清视频在线| 久久综合久久综合久久| 日韩一级在线观看| 最新亚洲电影| 亚洲人成人99网站| 亚洲精品国产欧美| 亚洲黄色免费| 亚洲日韩成人| 亚洲精品综合精品自拍| 久久精品国产一区二区三区| 在线观看视频免费一区二区三区 | 欧美www在线| 免费日本视频一区| 欧美高清视频一区二区| 亚洲四色影视在线观看| 狠狠久久亚洲欧美| 韩国女主播一区二区三区| 欧美国产三区| 欧美区日韩区| 国产精品进线69影院| 免费观看亚洲视频大全| 亚洲一区影院| 欧美一区二区三区免费视| 日韩视频在线你懂得| 卡一卡二国产精品| 欧美大片一区二区| 亚洲日本在线视频观看| 久久综合色天天久久综合图片| 中日韩美女免费视频网站在线观看| 国内成人在线| 亚洲精品永久免费精品| 一区免费观看| 日韩视频一区| 销魂美女一区二区三区视频在线| 亚洲精品乱码久久久久久黑人| 国产麻豆91精品| 影音先锋中文字幕一区| 国产日韩精品久久| 亚洲国产精品成人一区二区 | 午夜日本精品| 久久久欧美一区二区| 亚洲欧美中文日韩在线| aa级大片欧美| 久久高清福利视频| 欧美日本一区二区三区| 米奇777在线欧美播放| 久久精品亚洲精品国产欧美kt∨| 亚洲在线一区二区| 久久综合99re88久久爱| 久久久九九九九| 欧美日韩大片一区二区三区| 欧美高清免费| 国产乱子伦一区二区三区国色天香| 欧美日韩一级片在线观看| 欧美成人免费一级人片100| 久久久久久网站| 欧美在线免费观看| 欧美激情精品久久久久久| 免费看亚洲片| 亚洲自拍偷拍麻豆| 欧美精品在线网站| 一区二区在线观看av| …久久精品99久久香蕉国产| 伊人久久大香线蕉综合热线 | 激情一区二区| 亚洲男人的天堂在线aⅴ视频| 亚洲一级免费视频|