• <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>
            posts - 18,  comments - 104,  trackbacks - 0

            相信很多人都見過(guò)這種代碼

            1 template <class T>
            2 class Test
            3 {
            4 };
            5 
            6 int main()
            7 {
            8     Test<int> t;
            9 }

            這是類模板和模板類的最常見的使用了,但是想想看,見過(guò)這種寫法嗎?

            1 // 

            3 MyClass<std::string()> t;


            注意紅色的部分,我沒(méi)有寫錯(cuò),它的確可以編譯通過(guò)。這時(shí)候一串大大的疑問(wèn)產(chǎn)生了:怎么類的實(shí)例也可以做模板參數(shù)嗎?不是只有類型才能做模板參數(shù)么?難道。。。

            我一開始看到這個(gè)的時(shí)候,也想不明白,怎么會(huì)呢?難道是幻覺?我決定測(cè)試一下,于是我寫了下面一段代碼:
             1 template <class T>
             2 struct return_value
             3 {
             4     typedef T type;
             5 };
             6 
             7 int main()
             8 {
             9     test<std::string()>::type t;
            10 }

            編譯通過(guò)了,可以知道t的類型是:std::basic_string<char,std::char_traits<char>,std::allocator<char>> (void)
            很長(zhǎng)一串,寫簡(jiǎn)單點(diǎn)就是 std::string (void).
            這是個(gè)什么東西呢?如果是函數(shù)指針,怎么說(shuō)也要有個(gè)*呀,可是沒(méi)有。那這是什么呢?

            它是個(gè)函數(shù)類型,不是函數(shù)指針,函數(shù)指針是指向函數(shù)類型的指針,所以沒(méi)有*(感謝eXile)。

            我給個(gè)例子吧,相信大家對(duì)boost::function不陌生吧,看下面的代碼:

             1 std::string Test()
             2 {
             3     return "Hello World!";
             4 }
             5 
             6 boost::function<std::string()> func(Test);
             7 
             8 int main()
             9 {
            10     std::cout<<func();
            11 }

            Hello World應(yīng)該很熟悉了吧,而那個(gè)紅色的模板參數(shù)呢,其實(shí)不是調(diào)用std::string的構(gòu)造函數(shù),而是傳入std::string (void)(函數(shù)類型)。這個(gè)類型呢,沒(méi)啥用,不能實(shí)例化,也不能調(diào)用(不是函數(shù)指針),它只是作為一種類型的簽名而已。主要任務(wù)是告訴boost::function這個(gè)類,它將要接受的函數(shù)是一個(gè)接受void為參數(shù),返回std::string的一個(gè)函數(shù)而已,用來(lái)做類型推導(dǎo)。

            再看個(gè)例子吧,

             1 std::string Test(std::string str)
             2 {
             3     return "Hello " + str;
             4 };
             5 
             6 boost::function<std::string (std::string)>
             7     func(Test);
             8 
             9 int main()
            10 {
            11     std::cout<<func("World");
            12 }

            當(dāng)把簽名改成std::string (std::string)時(shí),就容易理解了。這個(gè)看似和函數(shù)聲明一樣的模板參數(shù),主要是為了讓boost::function做類型推導(dǎo)。順便加個(gè)括號(hào),看起來(lái)像函數(shù)。括號(hào)里面是參數(shù)的類型,括號(hào)外面是返回類型。

            這才是以函數(shù)指針作為模板類型的代碼:
             1 std::string Test(std::string str)
             2 {
             3     return "Hello " + str;
             4 }
             5 
             6 template <class T>
             7 class Type
             8 {
             9     typedef T type;
            10 };
            11 
            12 int main()
            13 {
            14     // use function pointor as a template argument, it's OK.
            15     Type<std::string (*)(std::string)>::type func = Test;
            16     // use the signature as a template argument, complier will tell you it's wrong.
            17     Type<std::string (std::string)>::type fun = Test;
            18 
            19 }

            最后總結(jié)一下,這種類型的模板參數(shù)只是作為簽名而存在,純粹是為了做類型推導(dǎo)。不能實(shí)例化,不是函數(shù)指針,名字叫:函數(shù)類型。

            posted on 2009-05-24 20:21 尹東斐 閱讀(2348) 評(píng)論(10)  編輯 收藏 引用

            FeedBack:
            # re: 類實(shí)例能做做模板參數(shù)嗎?
            2009-05-24 21:03 | hyrish
            模板參數(shù)分兩種,一種是數(shù)據(jù)類型,另一種是參數(shù)實(shí)例,這兩種本來(lái)就是C++標(biāo)準(zhǔn)里的,參數(shù)實(shí)例,相當(dāng)在聲明模板類實(shí)例時(shí),傳遞一個(gè)參數(shù),以提供一些模板類聲明時(shí)才能確定的參數(shù)。  回復(fù)  更多評(píng)論
              
            # re: 類實(shí)例能做做模板參數(shù)嗎?
            2009-05-25 09:28 | null
            @hyrish
            樓上理解錯(cuò)lz的意思了  回復(fù)  更多評(píng)論
              
            # re: 類實(shí)例能做做模板參數(shù)嗎?
            2009-05-25 19:08 | 尹東斐
            @hyrish

            你說(shuō)的那種是函數(shù)模板,函數(shù)模板的模板參數(shù)可以靠編譯器自動(dòng)推導(dǎo),以類型明確的變量為基礎(chǔ)。而在類模板中,只能事先聲明,然后使用,編譯器無(wú)法推導(dǎo)。  回復(fù)  更多評(píng)論
              
            # re: 類實(shí)例能做做模板參數(shù)嗎?[未登錄](méi)
            2009-05-27 10:14 | hdqqq
            std::string() 也可以看作是一個(gè)類型.

            long test(void);
            sizeof(test) 和 sizeof( test() )
            前面一個(gè)是 函數(shù)指針, 后面一個(gè)是 test 的返回值 long.

            你可以看看這個(gè)的輸出.
            std::cout << typeid(std::string).name() << std::endl;
            std::cout << typeid(std::string()).name() << std::endl;

            在下面這句話里
            boost::function<std::string()> func(Test);

            這個(gè)std::string 的變量根本不會(huì)被構(gòu)造, 所以不存在 所謂的實(shí)例.  回復(fù)  更多評(píng)論
              
            # re: 類實(shí)例能做做模板參數(shù)嗎?
            2009-05-27 14:04 | 尹東斐
            @hdqqq

            在模板的位置上,它的確是個(gè)類型,但是我不知道你想說(shuō)明什么?

            整篇文章,我都在嘗試說(shuō)明在模板參數(shù)的位置上,std::string()只是個(gè)標(biāo)記,純粹用作類型推導(dǎo),只是這種寫法很容易會(huì)誤導(dǎo)人讓人認(rèn)為那是個(gè)構(gòu)造函數(shù)。而這個(gè)誤導(dǎo)源自于 return type deduction,因?yàn)樵谛碌腃++0x中,可能會(huì)有

            int Test()
            {
            return 3;
            }

            type(Test()) i = 8;

            如果這個(gè)被實(shí)現(xiàn)的話(vs2010已經(jīng)有auto了,我想這個(gè)應(yīng)該不遠(yuǎn)了).編譯器可以做返回類型推導(dǎo),那么在模板參數(shù)的位置上,寫函數(shù)調(diào)用也就不足為奇了。

            所以,這篇文章主要想說(shuō)明,到目前為止,函數(shù)返回類型是推導(dǎo)不出來(lái),而類似std::string()的寫法只是個(gè)折中方案而已。

            當(dāng)然不是構(gòu)造函數(shù)調(diào)用。  回復(fù)  更多評(píng)論
              
            # re: 類實(shí)例能做做模板參數(shù)嗎?
            2009-05-27 15:43 | eXile
            基本概念錯(cuò)誤,LZ應(yīng)該了解一下什么是函數(shù)類型,什么是函數(shù)指針類型,他們之間的區(qū)別和轉(zhuǎn)化。  回復(fù)  更多評(píng)論
              
            # re: 類實(shí)例能做做模板參數(shù)嗎?[未登錄](méi)
            2009-06-02 13:38 | hdqqq
            typedef void (type_func)(void);
            typedef void (*ptype_func)(void);

            template <typename T>
            class template_class
            {
            private:
            typedef T local_type;
            public:
            template_class()
            {
            std::cout << typeid(local_type).name() << std::endl;
            }
            };

            void test()
            {
            template_class<int> la;
            template_class<type_func> lg;
            template_class<ptype_func> lb;
            template_class<int()> le;
            template_class<int(int)> lf;
            template_class<int(int())> lc;
            template_class<void()> ld;
            }

            上面的代碼可編譯通過(guò)  回復(fù)  更多評(píng)論
              
            # re: 類實(shí)例能做做模板參數(shù)嗎?
            2009-06-02 20:20 | 尹東斐
            @hdqqq

            樓上eXile說(shuō)了,這個(gè)類型是叫函數(shù)類型的,但是它不能被實(shí)例化。也就是你的type_func。  回復(fù)  更多評(píng)論
              
            # re: 類實(shí)例能做做模板參數(shù)嗎?[未登錄](méi)
            2009-06-03 21:08 | hdqqq
            是啊,所以我讓你看
            typeid(std::string()).name()
            的輸出.  回復(fù)  更多評(píng)論
              
            # re: 類實(shí)例能做做模板參數(shù)嗎?
            2009-06-04 23:52 | 尹東斐
            @hdqqq

            嗯,謝謝,領(lǐng)教了。  回復(fù)  更多評(píng)論
              

            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問(wèn)   Chat2DB   管理


            <2009年5月>
            262728293012
            3456789
            10111213141516
            17181920212223
            24252627282930
            31123456

            常用鏈接

            留言簿(4)

            隨筆檔案

            文章分類

            文章檔案

            相冊(cè)

            好友博客

            搜索

            •  

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            久久人人爽人人爽人人片AV不| 伊人久久大香线蕉成人| 亚洲日韩中文无码久久| 人人狠狠综合久久88成人| 久久99中文字幕久久| 久久久久99精品成人片三人毛片| 久久国产精品视频| 久久亚洲国产成人精品性色| 狠狠狠色丁香婷婷综合久久五月 | 久久久久久午夜精品| 国产成人精品久久| 国内精品久久久久久久久| 久久天天躁狠狠躁夜夜2020一| 精品乱码久久久久久久| 午夜精品久久久久久久无码| 久久发布国产伦子伦精品| 久久久久亚洲AV综合波多野结衣| 日产精品久久久一区二区| 免费一级做a爰片久久毛片潮| 国产精品久久久久久搜索| yy6080久久| 久久久久久久久久久免费精品| 国内精品久久久久影院优| 97精品依人久久久大香线蕉97 | 精品久久久久久中文字幕人妻最新 | 中文字幕无码免费久久| 久久露脸国产精品| 热久久这里只有精品| 国内精品九九久久久精品| 精品熟女少妇AV免费久久| 麻豆久久| 久久久久人妻一区二区三区 | 7777久久久国产精品消防器材| 人妻无码久久精品| 久久国产成人午夜AV影院| 精品免费久久久久国产一区| 色综合久久综精品| 国产精品久久久99| 久久精品国产欧美日韩| 国产精品美女久久久网AV| 久久青青草原精品国产不卡 |