• <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>

            Error

            C++博客 首頁(yè) 新隨筆 聯(lián)系 聚合 管理
              217 Posts :: 61 Stories :: 32 Comments :: 0 Trackbacks

            boost::implicit_cast

            在stackoverflow上看到這個(gè)帖子, 于是發(fā)現(xiàn)了boost::implicit_cast這個(gè)小東西.

            先來(lái)看看這段代碼:

            struct top {};
            struct mid_a : top {};
            struct mid_b : top {};
            struct bottom : mid_a, mid_b {};
            
            void foo(mid_a&) {}
            void foo(mid_b&) {}
            void bar(bottom &arg) {
                foo(arg); // 想要調(diào)用"void foo(mid_a&)"
            }
            
            int main() {
                bottom x;
                bar(x);
                return 0;
            }

            是無(wú)法編譯通過(guò)的, 因?yàn)閒oo的重載解析有歧義. 那么把bar里的代碼改一改, 為了保持C++風(fēng)格, 我們使用static_cast, 而不是C風(fēng)格的轉(zhuǎn)換:

            foo(static_cast<mid_a&>(arg));

            程序編譯通過(guò)了, 運(yùn)行起來(lái)也沒(méi)有問(wèn)題, 然而…

            一個(gè)月以后我把bar的參數(shù)類型修改了一下:

            struct top {};
            struct mid_a : top {};
            struct mid_b : top {};
            struct bottom : mid_a, mid_b {};
            
            void foo(mid_a&) {}
            void foo(mid_b&) {}
            void bar(top &arg) {
                // ... 過(guò)了一個(gè)月, 這里已經(jīng)添加了很多代碼.
                foo(static_cast<mid_a&>(arg));
            }
            
            int main() {
                top x;
                bar(x);
                return 0;
            }

            代碼依舊編譯通過(guò), 可是運(yùn)行時(shí)程序掛掉了(假設(shè)這幾個(gè)類里面有許多成員, 并且在foo里對(duì)其進(jìn)行了訪問(wèn)).

            發(fā)現(xiàn)問(wèn)題了嗎? 原因就在于static_cast太強(qiáng)大了, 強(qiáng)大到可以進(jìn)行”down-cast”. 于是編譯器沒(méi)有給你任何警告, 就把一個(gè)top類型的引用給強(qiáng)制轉(zhuǎn)換成了min_a的引用.

            這個(gè)時(shí)候輪到boost::implicit_cast出場(chǎng)了. 把bar里面的那句foo調(diào)用改一改:

            foo(boost::implicit_cast<mid_a&>(arg));

            于是一個(gè)月前的代碼依舊可以通過(guò)編譯, 而一個(gè)月后的代碼中的錯(cuò)誤被編譯器揪出來(lái)了. 原因在于隱式類型轉(zhuǎn)換不允許”down-cast”, 只能”up-cast”.

            這里簡(jiǎn)要說(shuō)一下所謂顯式和隱式類型轉(zhuǎn)換的區(qū)別. 在C++世界的英文里, 我們說(shuō)”convert”通常指”implicit convert”, 而”cast”指”explicit cast”. 隱式類型轉(zhuǎn)換好理解, 就是你寫了個(gè)a=b, 而ab不同類型, 編譯又不報(bào)錯(cuò), 就說(shuō)明隱式類型轉(zhuǎn)換發(fā)生了, 類似的情況還有在函數(shù)調(diào)用的參數(shù)傳遞時(shí). 而顯式類型轉(zhuǎn)換特指C風(fēng)格的強(qiáng)制轉(zhuǎn)換((type)obj或者C++中等價(jià)的type(obj)), 以及C++風(fēng)格的四個(gè)關(guān)鍵字(static_cast, const_cast, dynamic_cast, reinterpret_cast). 然而這個(gè)定義是相當(dāng)模糊的, 比如一個(gè)int類型的x, bool(x)是顯式的, 而!!x是隱式的, 其實(shí)效果上并沒(méi)有區(qū)別, 只是字面上的不同罷了. (關(guān)于cast和convert的區(qū)別, 參見(jiàn)這里這里)

            所以在bar里我們需要的僅僅是一個(gè)隱式類型轉(zhuǎn)換, 然而直接把a(bǔ)rg傳遞給foo的話會(huì)出現(xiàn)重載歧義, 于是我們需要告訴編譯器到底要進(jìn)行哪個(gè)隱式類型轉(zhuǎn)換. 然而static_cast又太過(guò)強(qiáng)大, 它還能做隱式類型轉(zhuǎn)換之外的事情(up-cast), 于是在日后代碼演化的過(guò)程中留下了bug.

            于是boost::implicit_cast應(yīng)運(yùn)而生, 它比static_cast弱, 正如它的名字一樣, 它只能用來(lái)告訴編譯器執(zhí)行什么隱式類型轉(zhuǎn)換.

            而它的代碼呢? 簡(jiǎn)單到令人發(fā)指:

            template <typename T>
            inline T implicit_cast (typename mpl::identity<T>::type x) {
                return x;
            }

            而mpl::identity的定義也極其簡(jiǎn)單:

            template<typename T> struct identity { typedef T type; };

            有人要問(wèn)這個(gè)identity干什么用的, 看起來(lái)很累贅. 如果沒(méi)有這個(gè)identity, 像”implicit_cast(obj)”這樣的代碼也能通過(guò)編譯, 然而它其實(shí)什么也沒(méi)做, obj的類型仍然沒(méi)變. identity的存在使得函數(shù)模板的參數(shù)類型推導(dǎo)失效, 因?yàn)橐茖?dǎo)出T, 首先得知道identity是什么, 而identity又是依賴于T的. 于是就形成了循環(huán)依賴, 參數(shù)類型推導(dǎo)就失效了. 于是編譯器就要求你顯式地指定T的類型.

            posted on 2012-06-23 21:34 Enic 閱讀(1691) 評(píng)論(1)  編輯 收藏 引用 所屬分類: 為什么用boost

            評(píng)論

            # re: (轉(zhuǎn))boost::implicit_cast 2014-11-04 15:33 ag
            分析的不錯(cuò)  回復(fù)  更多評(píng)論
              

            亚洲国产精品无码久久久蜜芽| 精品国产一区二区三区久久| 久久这里有精品| 一本色道久久HEZYO无码| 狠狠狠色丁香婷婷综合久久五月| 久久综合九色综合97_久久久| 99热都是精品久久久久久| 久久久久久亚洲精品影院| 蜜桃麻豆www久久| 99精品久久久久久久婷婷| 国产亚洲精久久久久久无码AV| 亚洲国产另类久久久精品小说 | 97r久久精品国产99国产精| 久久久久99精品成人片三人毛片 | 日韩一区二区久久久久久| av色综合久久天堂av色综合在 | 久久久久久国产精品无码下载 | 国产精品内射久久久久欢欢 | 国产免费久久精品99久久| 国产高潮国产高潮久久久| 亚洲中文字幕无码久久2020| 久久久久黑人强伦姧人妻| A级毛片无码久久精品免费| 狠狠88综合久久久久综合网| 亚洲香蕉网久久综合影视| 久久综合久久伊人| 久久久久久亚洲精品不卡| 亚洲嫩草影院久久精品| 99久久精品国产一区二区| 国产一级做a爰片久久毛片| 精品国产乱码久久久久久1区2区| 中文字幕热久久久久久久| 亚洲色婷婷综合久久| 亚洲精品蜜桃久久久久久| 99久久无色码中文字幕人妻| 麻豆成人久久精品二区三区免费 | 亚洲午夜久久久影院| 久久天堂AV综合合色蜜桃网| 久久精品国产亚洲AV无码娇色 | 国产精品成人久久久| 2021久久精品免费观看|