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

            woaidongmao

            文章均收錄自他人博客,但不喜標(biāo)題前加-[轉(zhuǎn)貼],因其丑陋,見(jiàn)諒!~
            隨筆 - 1469, 文章 - 0, 評(píng)論 - 661, 引用 - 0
            數(shù)據(jù)加載中……

            STL中的Concept和Boost庫(kù)的Concept_Check

            在generic programming當(dāng)中,一個(gè)重要的概念就是concept(滑稽的是,如果把這個(gè)concept也翻譯成“概念”,那就狗屁不通了。我傾向于說(shuō)它是“操作集”)。concept就是一組操作,如果一個(gè)type具有這些操作,那么就說(shuō)這個(gè)type是這個(gè)concept的一個(gè)model。
            這其中的思想有那么一丁點(diǎn)像是oo當(dāng)中的interface,一個(gè)class如果實(shí)現(xiàn)了一個(gè)interface,那么它就可以被當(dāng)作這個(gè)interface來(lái)用。同樣,如果一個(gè)type是一個(gè)concept的model,那么所有接受這個(gè)concept的操作也就可以接受這個(gè)type。
            例如,在stl中,stable_sort這個(gè)算法必須接受randomaccessiterator,這里randomaccessiterator就是一個(gè)concept,它規(guī)定自己的model必須可以進(jìn)行下標(biāo)運(yùn)算,那么不滿足這個(gè)concept的type就無(wú)法被編譯器接受(搞笑的是我在vc71里面把list<int> 的iterator傳給stable_sort,它居然欣然接受,要知道list的iterator應(yīng)該只是一個(gè)bidirectionaliterator 啊,比randomaccessiterator弱多了。不過(guò)仔細(xì)看看代碼,發(fā)現(xiàn)vc71的stable_sort接受bidirectionaliterator就夠了,不知道是好還是壞)。在gcc下面如果傳遞一個(gè)bidirectionaliterator給stable_sort,會(huì)得到一堆不知所云的錯(cuò)誤提示,讓人摸不著頭腦。
            c++語(yǔ)言本身并沒(méi)有對(duì)于concept的直接支持,stl解決這個(gè)問(wèn)題的辦法是用了一些traits來(lái)限制iterator的特性,以達(dá)到在編譯時(shí)期檢查concept的目的。但是traits導(dǎo)致的編譯錯(cuò)誤提示實(shí)在是太可怕了,我非常懷疑有哪個(gè)正常人可以從這些錯(cuò)誤提示推測(cè)出自己錯(cuò)在哪里。
            在boost庫(kù)里面,提供了一個(gè)conceptcheck庫(kù),它可以幫助我們寫出帶有concept檢查的代碼,而且沒(méi)有運(yùn)行時(shí)的開(kāi)銷,一旦用戶違反concept限制,輸出的錯(cuò)誤提示也比較好懂。舉個(gè)例子先,如果stl里面有conceptcheck,那么它的stable_sort大約會(huì)這樣:
            #include <boost/concept_check.hpp>
            template <class randomaccessiter>
            void stable_sort(randomaccessiter first, randomaccessiter last)
            {
                function_requires< randomaccessiteratorconcept<randomaccessiter> >();
                //... bla bla bla......
            }
            有了這個(gè) function_requires ,如果再傳遞給它list的iterator,編譯器(vc71)就會(huì)報(bào)這樣的錯(cuò):


            c:\boost_1_31_0\boost\concept_check.hpp(642): error c2676: 二進(jìn)制“+=” : “std::list<_ty>::iterator”不定義該運(yùn)算符或到預(yù)定義運(yùn)算符可接收的類型的轉(zhuǎn)換
                    with
                    [
                        _ty=int
                    ]
            當(dāng)然還有很多別的,但是至少它說(shuō)了一點(diǎn):傳入的iterator不滿足某個(gè)運(yùn)算。這對(duì)于用戶來(lái)說(shuō),應(yīng)當(dāng)是一個(gè)很有用的提示。
            使用concept check還有一個(gè)額外的好處,那就是調(diào)用一個(gè) function_requires可遠(yuǎn)比寫一些traits容易,而且代碼也清晰好維護(hù)。

            這個(gè)好用的 function_requires 就定義在concept_check.hpp當(dāng)中:
            template <class concept>
            inline void function_requires(mpl::identity<concept>* = 0)
            {
            #if !defined(ndebug)
              void (concept::*x)() = boost_fptr concept::constraints;
              ignore_unused_variable_warning(x);
            #endif
            }
            換句話說(shuō),function_requires只在debug中起作用,那么是不是在debug當(dāng)中它就添加了overhead呢?其實(shí)也沒(méi)有,仔細(xì)看看代碼:
            void (concept::*x)() = boost_fptr concept::constraints;
            這一句取constraints的地址。妙就妙在它讓編譯器“注意到”constraints,但又沒(méi)有真正調(diào)用constraints,而constraints是一個(gè)虛函數(shù),做實(shí)際的check。例如在randomaccessiteratorconcept(檢查iterator是否符合randomaccessiterator的concept)當(dāng)中,constraint是這個(gè)樣子:
            template <class tt>
            struct randomaccessiteratorconcept
            {
                void constraints() {
                function_requires< bidirectionaliteratorconcept<tt> >();
                function_requires< comparableconcept<tt> >();
            #ifndef boost_no_std_iterator_traits
                typedef typename std::iterator_traits<tt>::iterator_category c;
                function_requires< convertibleconcept< c,
                    std::random_access_iterator_tag> >();
                typedef typename std::iterator_traits<tt>::reference r;
            #endif

                i += n;             // require assignment addition operator
                i = i + n; i = n + i; // require addition with difference type
                i -= n;             // require assignment subtraction operator
                i = i - n;                  // require subtraction with difference type
                n = i - j;                  // require difference operator
                (void)i[n];                 // require element access operator
                }
                tt a, b;
                tt i, j;
            #ifndef boost_no_std_iterator_traits
                typename std::iterator_traits<tt>::difference_type n;
            #else
                std::ptrdiff_t n;
            #endif
            };
            從這個(gè)實(shí)現(xiàn)中我們完全可以讀出randomaccessiterator的具體含義:
            1. 它必須是一個(gè)bidirectionaliterator
            2. 它必須滿足“可比較 (comparable)”的concept
            3. 它還必須滿足“可轉(zhuǎn)換 (convertible)”的concept,而且是轉(zhuǎn)換成自己的iterator_category類別

            4. 它必須定義了reference這個(gè)type
            5. 這是最重要的,它必須有difference_type,而且可以進(jìn)行 +, -, +=, -= 的運(yùn)算
            還要記得,由于這個(gè)函數(shù)沒(méi)有真正的被調(diào)用過(guò),所以無(wú)論你怎么寫,它都不會(huì)變成實(shí)際的代碼,所以也不會(huì)影響運(yùn)行效率的!

            posted on 2008-07-26 23:08 肥仔 閱讀(1113) 評(píng)論(0)  編輯 收藏 引用 所屬分類: C++ 基礎(chǔ)

            亚洲国产另类久久久精品 | 香蕉99久久国产综合精品宅男自| 91精品国产综合久久四虎久久无码一级| 久久免费小视频| 一本色综合久久| 久久福利青草精品资源站免费| 三级片免费观看久久| 国产午夜福利精品久久2021| 久久涩综合| 潮喷大喷水系列无码久久精品| 天堂无码久久综合东京热| 精品久久久无码人妻中文字幕豆芽| 久久精品视屏| 国产精品久久久久9999高清| 青春久久| 精品国产婷婷久久久| 久久青青草原综合伊人| 嫩草伊人久久精品少妇AV| 色诱久久av| 久久精品女人天堂AV麻| 精品国产91久久久久久久a| 999久久久无码国产精品| 色综合久久无码中文字幕| 日韩精品久久久久久久电影| 久久精品中文字幕第23页| 久久精品无码一区二区三区| 国产V综合V亚洲欧美久久| 国产精品9999久久久久| 97久久精品无码一区二区| 九九精品99久久久香蕉| 看久久久久久a级毛片| 青青草原精品99久久精品66| 久久亚洲精品国产精品| 久久久久久久精品妇女99| 亚洲精品乱码久久久久久中文字幕| 性高朝久久久久久久久久| 少妇被又大又粗又爽毛片久久黑人 | 日韩电影久久久被窝网| 久久精品国产欧美日韩| 久久久久无码国产精品不卡| 中文精品99久久国产|