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

Impossible is nothing  
  愛過知情重醉過知酒濃   花開花謝終是空   緣份不停留像春風來又走   女人如花花似夢
公告
日歷
<2025年9月>
31123456
78910111213
14151617181920
21222324252627
2829301234
567891011
統(tǒng)計
  • 隨筆 - 8
  • 文章 - 91
  • 評論 - 16
  • 引用 - 0

導航

常用鏈接

留言簿(4)

隨筆分類(4)

隨筆檔案(8)

文章分類(77)

文章檔案(91)

相冊

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

 

看C++ Templates 16.1 Named Template Arguments

書中的例子實現(xiàn)手法使用多重/虛擬繼承, 實現(xiàn)手法感覺比較詭秘. 但是至少告訴我是可以實現(xiàn)的.

于是干脆自己也練了練手,  博君一笑. 只在VC7.1下測試過, VC6也許可能可以迂回實現(xiàn), 但是估計工作量太大.

1. 首先需要一個基本的 If 語句.

template <bool, class T, class U>
struct if_
{
    typedef T type;
};

template<class T, class U>
struct if_<false,  T,  U>
{
    typedef U type;
};

2. 然后使用一個 type_selector meta-function, N表示第幾個默認參數(shù)(注意我的默認Policy參數(shù)DefaultPolicyArgs里面有一個meta data, 為0. 如果是用戶定義的Policy, 那么形如Policy2_is的模板類里面有一個meta data為2. 這個數(shù)字主要是用于定位.

最后的DefaultType是當掃描一遍, 發(fā)現(xiàn)沒有任何對應(yīng)N位置的自定義Policy參數(shù), 那么就取這個為默認值, 結(jié)束遞歸.(下面的4個void的特化版本就是干這個的)

template<
    int N,
    class T1,
    class T2,
    class T3,
    class T4,
    class DefaultType>
struct type_selector
{
    typedef typename if_ <
        (T1::value == N),
        T1,
        type_selector<N, T2, T3, T4, void, DefaultType> >::type eval_type;

    typedef typename eval_type::type type;
};

//shift以后, 如果都是默認值, 遞歸會來到這里, 結(jié)束.
template<
    int N,
    class DefaultType>
struct type_selector<N, void, void, void, void, DefaultType>
{
    typedef DefaultType type;
};

struct DefaultPolicy1 {};
struct DefaultPolicy2 {};
struct DefaultPolicy3 {
  public:
    static void doPrint() {
        std::cout << "DefaultPolicy3::doPrint()\n";
    }
};
class DefaultPolicy4 {};

struct  DefaultPolicyArgs {
    static const int value = 0;
};


template <typename Policy>
struct Policy1_is
{
    typedef Policy type;
    static const int value = 1;
};

template <typename Policy>
struct Policy2_is
{
    typedef Policy type;
    static const int value = 2;
};


template <typename Policy>
struct Policy3_is
{
    typedef Policy type;
    static const int value = 3;
};


template <typename Policy>
struct Policy4_is
{
    typedef Policy type;
    static const int value = 4;
};


template<class T1, class T2, class T3, class T4>
struct PolicySelector
{
    typedef typename type_selector<1, T1, T2, T3, T4, DefaultPolicy1>::type P1;
    typedef typename type_selector<2, T1, T2, T3, T4, DefaultPolicy2>::type P2;
    typedef typename type_selector<3, T1, T2, T3, T4, DefaultPolicy3>::type P3;
    typedef typename type_selector<4, T1, T2, T3, T4, DefaultPolicy4>::type P4;
};
    

template <typename T1 = DefaultPolicyArgs,
          typename T2 = DefaultPolicyArgs,
          typename T3 = DefaultPolicyArgs,
          typename T4 = DefaultPolicyArgs>
class BreadSlicer {
    typedef typename PolicySelector<T1, T2, T3, T4> Policies;
    
  public:
    void print () {
        std::cout << typeid(Policies::P3).name() << std::endl;
        Policies::P3::doPrint();
    }
    void  print_2()
    {
        std::cout << typeid(Policies::P2).name() << std::endl;
        Policies::P2::print_2();
    }
    //...
};


//下面的就是測試代碼了.

class CustomPolicy {
  public:    
    static void doPrint() {
        std::cout << "CustomPolicy::doPrint()\n";
    }
};

class CustomPolicy2 {
public:
    static void print_2()
    {
        std::cout << "Here is CustomPolicy2 instance" << std::endl;
    }
};
    
int main()
{
    BreadSlicer<> bc1;
    bc1.print();

    BreadSlicer< Policy3_is<CustomPolicy>,
        Policy2_is<CustomPolicy2> > bc2;
    
    bc2.print();
    return 0;
}

上面那個帖子的實現(xiàn)手法不太好, 當client使用的時候, 還是需要

 BreadSlicer< Policy3_is<CustomPolicy>, Policy2_is<CustomPolicy2> > bc;

復雜的嵌套模板語法, 如果能夠去掉PolicyN_is, 例如可以這樣

 BreadSlicer<> bc2;  //全部默認policies

如果定制其中的policy2, 這樣
 BreadSlicer< CustomPolicy2 > bc;

如果需要定制2, 3, 這樣

 BreadSlicer< CustomPolicy2, CustomPolicy3 > bc;

與順序無關(guān), 先寫3, 再寫2也可以
 BreadSlicer< CustomPolicy3, CustomPolicy2 > bc;

那就更加簡單了.

幸運的是, 這也是可以實現(xiàn)的, 而且與前面的帖子相比, 這個新的實現(xiàn)還直白簡單, 使用起來由于直接使用Policy class作為參數(shù),
而無需通過PolicyN_is這樣的包裹, 使用起來也更加優(yōu)雅.

還是看看代碼:

1. 同上, 定義一個if語句.

template <bool, class T, class U>
struct if_
{
    typedef typename T type;
};

template<class T, class U>
struct if_<false,  T,  U>
{
    typedef typename U type;
};

2. 定義一個wrapper, 使得 type_wrapper<T>::type 有效(為T). 因為直接使用 T::type 可能遇到T根本沒有type這個typedef內(nèi)嵌類型.

template<class T>
struct type_wrapper
{
    typedef T type;
};

struct  DefaultPolicyArgs {
    static const int value = 0;            //特殊meta-data, 為0表示默認參數(shù)
};


3. 然后使用一個 type_selector meta-function, N表示第幾個默認參數(shù)(注意我的默認Policy參數(shù)DefaultPolicyArgs里面有一個meta data, 為0. 如果是用戶定義的Policy, 那么它也應(yīng)該定義一個meta data,. 這個數(shù)字主要是用于告訴selector它是想覆蓋第幾個默認的policy參數(shù).

最后的DefaultType是當掃描一遍, 發(fā)現(xiàn)沒有任何對應(yīng)N位置的自定義Policy參數(shù), 那么就取這個為默認值, 結(jié)束遞歸.(下面的4個DefaultPolicyArgs的特化版本就是干這個的)

與前面的一個版本相比, 我不再使用void, 而是使用DefaultPolicyArgs來填充, 這樣在大部分情況下匹配速度要快. (指編譯速度)

template<
    int N,
    class T1,
    class T2,
    class T3,
    class T4,
    class DefaultType>
struct type_selector
{
    typedef typename if_ <
        (T1::value == N),
        type_wrapper<T1>,
        type_selector<N, T2, T3, T4, DefaultPolicyArgs, DefaultType> >::type eval_type;

    typedef typename eval_type::type type;
};

//shift以后最終來到這里, 結(jié)束遞歸
template<
    int N,
    class DefaultType>
struct type_selector<N, DefaultPolicyArgs, DefaultPolicyArgs, DefaultPolicyArgs, DefaultPolicyArgs, DefaultType>
{
    typedef DefaultType type;
};


struct DefaultPolicy1 {};
struct DefaultPolicy2 {};
struct DefaultPolicy3 {
  public:
    static void doPrint() {
        std::cout << "DefaultPolicy3::doPrint()\n";
    }
};
class DefaultPolicy4 {};


template<class T1, class T2, class T3, class T4>
struct PolicySelector
{
    typedef typename type_selector<1, T1, T2, T3, T4, DefaultPolicy1>::type P1;
    typedef typename type_selector<2, T1, T2, T3, T4, DefaultPolicy2>::type P2;
    typedef typename type_selector<3, T1, T2, T3, T4, DefaultPolicy3>::type P3;
    typedef typename type_selector<4, T1, T2, T3, T4, DefaultPolicy4>::type P4;
};
   


template <typename T1 = DefaultPolicyArgs,
          typename T2 = DefaultPolicyArgs,
          typename T3 = DefaultPolicyArgs,
          typename T4 = DefaultPolicyArgs>
class BreadSlicer {
    typedef typename PolicySelector<T1, T2, T3, T4> Policies;
   
  public:
    void print () {
        std::cout << typeid(Policies::P3).name() << std::endl;
        Policies::P3::doPrint();
    }
    void  print_2()
    {
        std::cout << typeid(Policies::P2).name() << std::endl;
        Policies::P2::print_2();
    }
    //...
};


class CustomPolicy2 {
public:
    static const int value = 2;     //關(guān)鍵的修改在此, 這個是實現(xiàn)定制Policy時需要提供的meta data
    static void print_2()
    {
        std::cout << "Here is CustomPolicy2 instance" << std::endl;
    }
};

class CustomPolicy3 {
  public:   
    static const int value = 3;     //meta data, 同上, 3表示這個代表的是用了替換Policy3
    static void doPrint() {
        std::cout << "CustomPolicy3::doPrint()\n";
    }
};


//這樣, PolicyN_is 就沒有了, 唯一的要求就是, 當實現(xiàn)custom policy的時候, 別忘了在其中定義一個叫做
value的整形常量, N代表替換哪個默認的policy參數(shù).

int main()
{
    BreadSlicer<> bc1;                                 //全部默認
    bc1.print();

    BreadSlicer< CustomPolicy2, CustomPolicy3 > bc2;   //2,3定制, 是不是干凈一些?
   
    bc2.print();
    bc2.print_2();
    std::cout << std::flush;
    return 0;
}
posted on 2006-02-27 23:14 笑笑生 閱讀(357) 評論(0)  編輯 收藏 引用 所屬分類: C++語言
 
Copyright © 笑笑生 Powered by: 博客園 模板提供:滬江博客
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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人片在线观看桃| 久久久999| 欧美综合国产| 久久亚洲精品欧美| 久久久久免费视频| 亚洲第一页在线| 嫩草国产精品入口| 亚洲国产精品视频一区| 亚洲精品在线观看免费| 亚洲免费高清| 欧美中文字幕第一页| 久久婷婷亚洲| 国产精品xnxxcom| 国产亚洲欧美日韩精品| 亚洲欧洲在线免费| 性欧美videos另类喷潮| 免费日韩成人| 亚洲综合色视频| 欧美黄色日本| 亚洲福利视频免费观看| 欧美在线观看视频一区二区三区| 欧美日韩国产经典色站一区二区三区| 国产精品久久久久久久久久久久久久| 国产欧美日韩专区发布| 亚洲午夜久久久久久久久电影网| 久久视频精品在线| 午夜精品福利一区二区三区av| 欧美成人第一页| 狠狠久久婷婷| 欧美在线亚洲| 亚洲欧美日韩综合| 国产精品一区毛片| 欧美有码视频| 欧美一区午夜视频在线观看| 国产精品久久久久一区二区三区共| 91久久中文字幕| 亚洲电影中文字幕| 免费久久99精品国产| 亚洲精品国产精品国自产在线 | 亚洲裸体在线观看| 免费永久网站黄欧美| 日韩五码在线| 亚洲先锋成人| 在线免费观看成人网| 久久精品人人做人人综合| 欧美伊人久久久久久午夜久久久久 | 日韩视频免费观看| 亚洲大片一区二区三区| 欧美国产日韩精品免费观看| 这里只有视频精品| 久久www免费人成看片高清| 久久久久久久尹人综合网亚洲| 欧美亚日韩国产aⅴ精品中极品| 亚洲国产女人aaa毛片在线| 母乳一区在线观看| 欧美激情一区二区| 久久狠狠婷婷| 裸体女人亚洲精品一区| 中国av一区| 久久国产精品久久久| 亚洲精品在线三区| 欧美在线免费| 一区二区三区国产精华| 亚洲永久视频| 99re6这里只有精品| 性欧美超级视频| 中文无字幕一区二区三区| 亚洲欧美日韩综合国产aⅴ| 亚洲成在人线av| 亚洲免费在线视频一区 二区| 国产一区二区三区无遮挡| 中文网丁香综合网| 亚洲另类黄色| 欧美国产免费| 欧美成人免费全部观看天天性色| 国产精品久久久久久久久久久久| 欧美大片在线看免费观看| 亚洲欧美另类国产| 国产欧美日韩一区二区三区在线 | 久久国产精品免费一区| 欧美手机在线视频| 亚洲九九爱视频| 亚洲一区二区三区在线视频| 欧美日韩精品欧美日韩精品一| 亚洲国产三级| 欧美激情一区二区三区在线视频观看| 欧美一级二级三级蜜桃| 国产精品视频福利| 欧美一区二区三区另类| 久久综合国产精品| 亚洲日本免费| 国产伦精品一区二区三区高清| 久久久久久久久综合| 亚洲成色最大综合在线| 亚洲欧美成人在线| 亚洲精品一区中文| 国产精品www| 欧美激情国产日韩| 久久国产黑丝| 99这里只有精品| 免费永久网站黄欧美| 日韩一区二区精品| 国产综合网站| 国产亚洲欧美日韩日本| 欧美伦理一区二区| 久久婷婷综合激情| 欧美伊人久久久久久久久影院| 亚洲激情电影在线| 久久综合色88| 久久xxxx精品视频| 欧美中文日韩| 欧美在线亚洲综合一区| 亚洲欧美日本视频在线观看| 一区二区精品国产| 亚洲男人的天堂在线| 99精品热6080yy久久| 亚洲每日更新| 亚洲五月六月| 先锋影院在线亚洲| 销魂美女一区二区三区视频在线| 亚洲视频成人| 欧美一区二区三区成人| 久久久久久久一区| 久久久久99精品国产片| 亚洲国产成人精品久久| 亚洲国产精品va在线观看黑人| 欧美日韩一区二区在线观看视频| 欧美日韩精品免费在线观看视频| 欧美看片网站| 欧美视频日韩| 影院欧美亚洲| 午夜精品久久久久影视| 久久亚洲精品伦理| 亚洲人成在线播放网站岛国| 亚洲毛片av在线| 欧美一区二区黄| 欧美激情视频一区二区三区免费 | 欧美精品久久久久久久免费观看| 美国成人直播| 欧美揉bbbbb揉bbbbb| 国产一区二区三区的电影 | 久久久水蜜桃av免费网站| 欧美福利精品| 国产一区二区精品| 欧美一区2区视频在线观看| 久久综合九色| 亚洲欧美日韩国产| 国产精品欧美精品| 亚洲一区二区三区乱码aⅴ| 亚洲高清色综合| 欧美精品色一区二区三区| 在线观看日韩精品| 久久aⅴ国产紧身牛仔裤| 亚洲精品久久久久久久久久久久| 日韩一区二区精品| 欧美日韩色一区| 亚洲一二区在线| 亚洲一区二区动漫| 亚洲欧美日韩中文视频| 国产日韩欧美在线视频观看| 欧美一区二区三区日韩视频| 亚洲一区二区不卡免费| 国产日韩在线视频| 久久精彩免费视频| 美脚丝袜一区二区三区在线观看| 亚洲第一页自拍| 亚洲美女网站| 国产精品电影网站| 久久久亚洲国产天美传媒修理工| 久久青草欧美一区二区三区| 亚洲天堂成人在线观看| 欧美一级淫片aaaaaaa视频| 亚洲精品欧美日韩专区| 午夜在线视频观看日韩17c| 日韩视频亚洲视频| 久久精品一区二区国产| 亚洲欧美在线磁力| 欧美国产日韩亚洲一区| 欧美一区二区视频在线观看| 美女主播一区| 久久久www成人免费无遮挡大片| 一本综合精品| 亚洲国产天堂久久综合网| 亚洲天堂av高清| 日韩视频专区| 欧美激情视频免费观看| 久久亚洲精选| 国模精品一区二区三区色天香 | 国产精品人人爽人人做我的可爱 | 一区二区免费在线视频| 久久精品首页| 免费不卡亚洲欧美| 国产精品日产欧美久久久久| 久久成人18免费观看| 欧美欧美在线| 久久久久五月天| 欧美激情亚洲| 久久久久天天天天| 欧美日韩一区二区三区免费|