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

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

導航

常用鏈接

留言簿(4)

隨筆分類(4)

隨筆檔案(8)

文章分類(77)

文章檔案(91)

相冊

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

 

看C++ Templates 16.1 Named Template Arguments

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

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

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表示第幾個默認參數(注意我的默認Policy參數DefaultPolicyArgs里面有一個meta data, 為0. 如果是用戶定義的Policy, 那么形如Policy2_is的模板類里面有一個meta data為2. 這個數字主要是用于定位.

最后的DefaultType是當掃描一遍, 發現沒有任何對應N位置的自定義Policy參數, 那么就取這個為默認值, 結束遞歸.(下面的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以后, 如果都是默認值, 遞歸會來到這里, 結束.
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;
}

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

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

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

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

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

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

 BreadSlicer< CustomPolicy2, CustomPolicy3 > bc;

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

那就更加簡單了.

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

還是看看代碼:

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內嵌類型.

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

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


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

最后的DefaultType是當掃描一遍, 發現沒有任何對應N位置的自定義Policy參數, 那么就取這個為默認值, 結束遞歸.(下面的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以后最終來到這里, 結束遞歸
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;     //關鍵的修改在此, 這個是實現定制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 就沒有了, 唯一的要求就是, 當實現custom policy的時候, 別忘了在其中定義一個叫做
value的整形常量, N代表替換哪個默認的policy參數.

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>
            国产精品久久国产三级国电话系列| 亚洲精品在线视频| 欧美日韩一区二区三区在线 | 中文亚洲视频在线| 欧美不卡激情三级在线观看| 亚洲欧美激情精品一区二区| 欧美天堂亚洲电影院在线观看| 亚洲人成7777| 欧美成人黑人xx视频免费观看| 久久九九全国免费精品观看| 欧美综合77777色婷婷| 国产日韩欧美成人| 久久久久九九九九| 欧美中文字幕| 亚洲精品乱码久久久久久日本蜜臀| 午夜精品一区二区三区电影天堂 | 欧美天堂亚洲电影院在线观看| 一区二区三区产品免费精品久久75 | 亚洲激情午夜| 欧美刺激午夜性久久久久久久| 久久精品国产一区二区电影 | 亚洲桃色在线一区| 国产精品呻吟| 久久久久久久网| 亚洲午夜国产成人av电影男同| 欧美日韩三级在线| 亚洲免费视频在线观看| 亚洲一区二区成人| 国内精品久久久久久久果冻传媒| 久久只精品国产| 免费视频一区| 一区二区动漫| 午夜精品福利一区二区蜜股av| 国内揄拍国内精品久久| 亚洲大黄网站| 欧美午夜精品久久久久久浪潮| 新67194成人永久网站| 亚洲免费伊人电影在线观看av| 亚洲片在线资源| 国产精品免费看久久久香蕉| 欧美在线首页| 欧美成人免费大片| 老司机精品视频网站| 国产精品三级久久久久久电影| 亚洲国产综合91精品麻豆| 国产精品网站视频| 性欧美超级视频| 最新69国产成人精品视频免费| 免费一区视频| 欧美精品一区二区久久婷婷| 亚洲一区在线免费观看| 欧美在线日韩在线| 日韩视频精品在线观看| 午夜精品久久久久久久久久久久久| 国产午夜精品在线观看| 亚洲国产欧美久久| 国产欧美日韩综合一区在线观看 | 久久精品一二三区| 欧美成人伊人久久综合网| 亚洲欧美日本在线| 美女尤物久久精品| 欧美亚洲在线观看| 欧美激情第9页| 久久久久欧美精品| 欧美视频在线视频| 欧美国产欧美亚洲国产日韩mv天天看完整 | 欧美刺激性大交免费视频 | 久久国产一区二区三区| 欧美a一区二区| 欧美一级播放| 久久综合狠狠| 夜夜狂射影院欧美极品| 国产精品高潮在线| 久久综合狠狠综合久久激情| 美玉足脚交一区二区三区图片| 狠狠入ady亚洲精品| 欧美亚洲一区二区在线观看| 欧美freesex交免费视频| 99国产麻豆精品| 欧美三级电影网| 国产精品尤物| 宅男噜噜噜66国产日韩在线观看| 激情综合中文娱乐网| 亚洲免费成人av| 亚洲国产精品www| 欧美在线日韩| 久久精品女人| 国产农村妇女精品| 一区二区三区蜜桃网| 亚洲国产精品福利| 久久精品国产精品亚洲精品| 欧美一级日韩一级| 久久综合色天天久久综合图片| 亚洲国产精品热久久| 久久久久国产一区二区| 久久久水蜜桃| 国产一区在线视频| 欧美在线看片| 老色鬼精品视频在线观看播放| 国产日韩欧美亚洲一区| 性久久久久久| 久久琪琪电影院| 国产综合色一区二区三区 | 久久综合五月| 欧美激情亚洲激情| 亚洲三级网站| 欧美日本国产在线| 99国产一区| 午夜日韩激情| 国产伊人精品| 久色婷婷小香蕉久久| 亚洲精品美女久久久久| 日韩午夜精品| 国产精品久久久久久久久久三级 | 欧美成在线观看| 激情久久久久久| 在线一区视频| 欧美一区二区三区免费视| 久久精品一二三区| 亚洲欧美怡红院| 久久天天狠狠| 最新高清无码专区| 久久综合久色欧美综合狠狠| 国产精品美女久久久久久2018| 一区福利视频| 日韩亚洲一区在线播放| 亚洲欧美国产毛片在线| 亚洲高清在线观看| 亚洲人体偷拍| 免费在线视频一区| 国产日韩视频一区二区三区| 亚洲日韩视频| 欧美国产欧美亚州国产日韩mv天天看完整| 99re6这里只有精品| 欧美日韩午夜在线视频| 先锋影音一区二区三区| 欧美激情精品久久久久久大尺度| 亚洲神马久久| 精品白丝av| 欧美日韩色综合| 久久深夜福利免费观看| 一本色道久久综合狠狠躁篇怎么玩| 欧美一级久久久| 久久成人综合网| 西瓜成人精品人成网站| 欧美成人综合一区| 亚洲成人自拍视频| 欧美一区二区三区免费视频| 亚洲精品一区二区三| 国产日韩欧美在线| 欧美日韩在线亚洲一区蜜芽| 久久久久久久久久久成人| 一区二区日韩精品| 欧美激情成人在线| 久久国产66| 一区二区不卡在线视频 午夜欧美不卡'| 国产精品夜夜嗨| 欧美日韩国产三区| 蜜臀av在线播放一区二区三区| 亚洲女性裸体视频| 99ri日韩精品视频| 亚洲国产精品尤物yw在线观看| 久久久国产91| 午夜精品成人在线| 亚洲视频在线免费观看| 亚洲精品麻豆| 亚洲国产精品第一区二区三区| 国产午夜精品在线| 国产区精品在线观看| 欧美日韩麻豆| 欧美精品福利| 欧美国产日韩一区二区三区| 久久综合九色欧美综合狠狠| 欧美在线一区二区三区| 欧美专区日韩专区| 亚洲精品中文在线| 亚洲欧美日韩精品久久久| 99视频精品| 日韩午夜激情av| 亚洲精品国产品国语在线app | 国产精品最新自拍| 久久久久久网址| 午夜在线a亚洲v天堂网2018| 免费成人av在线| 亚洲免费高清视频| 免费日韩视频| 欧美成人蜜桃| 中国成人在线视频| 99精品热视频只有精品10| 欧美国产综合视频| 亚洲美女视频网| 日韩视频在线观看免费| 国产精品欧美日韩| 蜜桃av一区二区三区| 亚洲一区二区三区在线| 亚洲第一精品久久忘忧草社区| 国产精品video| 久久久亚洲人| 亚洲欧美日本伦理| 亚洲精品一区二区网址 |