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

            西城

            指尖代碼,手上年華

            聯系 聚合 管理
              20 Posts :: 0 Stories :: 62 Comments :: 0 Trackbacks
            bind實質上是一個函數。
            #ifndef BOOST_BIND
            #define BOOST_BIND bind
            #endif
            在此文件內將其定義為BOOST_BIND.以下均以BOOST_BIND代指bind.

             

            template<class R, class F>
                _bi::bind_t<R, F, _bi::list0>
                BOOST_BIND(F f)
            {
             typedef _bi::list0 list_type;
             return _bi::bind_t<R, F, list_type> (f, list_type());
            }

             

            此函數的返回值是_bi::bind_t<R,F,_bi::list0>,參數是F,這是針對無參函數的版本。
            _bi是文件中定義于boost命名空間之下的一個命名空間,bind實現的細節均在其中(bind implementation).
            下面看bind_t定義:

             

            template<class R, class F, class L> class bind_t
            {
            public:

             typedef bind_t this_type;

             bind_t(F f, L const & l): f_(f), l_(l) {}

            #define BOOST_BIND_RETURN return
            #include <boost/bind/bind_template.hpp>
            #undef BOOST_BIND_RETURN
            };

             

            看起來很小,只有一個類型定義和一個構造函數。但注意include那一句,這意味著整個bind_template.hpp文件都
            是bind_t類的一部分。而bind_tmplate文件差不多有350行。因為函數體都很小,所以直接在類內定義,成為inline.所以,當要定義的一個類特別大時,將其具體的函數
            實現放置于另一個文件中也是一個很好的辦法。
            下面來看bind_template.hpp中屬于_bi::bind_t的一些成員定義。

            typedef typename result_traits<R, F>::type result_type; result_traits
            提取返回值。在bind.hpp中將其定義為R。

                result_type operator()()
                {
                    list0 a;
                    BOOST_BIND_RETURN l_(type<result_type>(), f_, a, 0);
                }
            后面依次是operator()的const版本,一直到九個參數的版本。BOOST_BIND_RETURN可以直接當作return.
            list0類也有九個。


            看一個例子
            int f()
            {
                return 1;
            }
            bind(f)()應該等于f().
            bind函數中返回值是:
            _bi::bind_t<R, F, list_type> (f, list_type());list_type=_bi::list0。所以返回值是一個特化的模板類。
            f的類型為int (*)(),即是F。所以返回值是
            _bi::bind_t<intint(*)(),_bi::list0>(f,_bi::list0());

             R為萃取出的返回值(int)。通過此構造函數,構造出了一個bind_t類,f,list0,分別由bind_t的相應私有數據存儲起來。
            然后是調用此類(函數體).即bind(f)(),bind_t重載了相應的()運算符。
            還是依參數個數不同有不同的表現形式,此例沒有參數,其表現形式為:

                result_type operator()() 
                {
                    list0 a;
                    BOOST_BIND_RETURN l_(type<result_type>(), f_, a, 0);
                }
            返回值即為原來的int.函數實現這一塊交給list0(l_).list0(type<int>(),f,a,0).
            type<int>()生成了一個空的類。l_已經是一個生成的類,所以此處調用的是重載操作符().
                template<class R, class F, class A> R operator()(type<R>, F & f, A &, long)
                {
                    return unwrapper<F>::unwrap(f, 0)();
                }

            又調用unwrapper結構體。因為是static,所以不用再生成類的實例,最后返回的就是原來的那個f()------->int。
            template<class F> struct unwrapper
            {
                static inline F & unwrap( F & f, long )
                {
                    return f;
                }

                template<class F2> static inline F2 & unwrap( reference_wrapper<F2> rf, int )
                {
                    return rf.get();
                }

                template<class R, class T> static inline _mfi::dm<R, T> unwrap( R T::* pm, int )
                {
                    return _mfi::dm<R, T>( pm );
                }
            };

            其他參數不同的情況與此類似。bind對不同數目的參數都有相應一致的處理方式。但是依據相應類的個數以及bind的重載個數,參數值最多只能有9個。這樣限制一般也
            沒什么大的影響,因為一般的參數個數沒有這么多。如果用的參數有太多的話,其實可以在源碼上再加上一種重載形式即可。

            上面的實現繞了這么大的一圈,其實最后調用的還是原來的那個函數,看似費時,其實都是為了泛型的必要。bind能夠綁定的函數類型大大地增加,不管是普通的函數
            指針,還是函數體,以及沒有result_type的類型,bind都可以很好的運作。而且還可以與ref庫結合起來使用。

            在C++的這些第三方庫里面,BOOST是比較特別的一個。因為它并不是專注于一個問題,而是涉及到了語言的各個層面。有很多接近于語言底層的特性,所以BOOST庫比
            任何一個庫都更值得我們去研究,揣摩。

             

            posted on 2012-04-02 11:38 西城 閱讀(2657) 評論(2)  編輯 收藏 引用 所屬分類: C/C++Boost

            Feedback

            # re: Boost::bind源碼分析:bind.hpp 2012-04-03 19:47 空明流轉
            其實很多東西揣摩了沒啥用處。
            有些就是語言機制的缺陷。
            比方說static assert,比方說這么多轉發。
            Varidic Template和Perfect Forward一出來了,很多tricky都失效了。。。  回復  更多評論
              

            # re: Boost::bind源碼分析:bind.hpp 2012-04-03 22:25 墨魂
            @空明流轉
            這樣至少用的時候知道是怎么回事。又有那個語言是完美的呢?都是人設計的,都會有缺陷。自己盡為吧。  回復  更多評論
              

            久久婷婷五月综合色奶水99啪| 久久97久久97精品免视看| 国产精品亚洲综合专区片高清久久久 | 亚洲v国产v天堂a无码久久| 欧美午夜精品久久久久久浪潮| 少妇精品久久久一区二区三区| 麻豆精品久久精品色综合| 国产成人久久精品麻豆一区| 中文成人无码精品久久久不卡| 99久久人妻无码精品系列蜜桃| 久久亚洲国产成人影院网站| 亚洲AV日韩精品久久久久| 999久久久免费国产精品播放| 嫩草伊人久久精品少妇AV| 久久精品国产WWW456C0M| 日日噜噜夜夜狠狠久久丁香五月 | 99久久精品国内| 色诱久久久久综合网ywww| 久久免费香蕉视频| 青青青国产精品国产精品久久久久 | 日韩人妻无码精品久久免费一 | 久久久久久国产a免费观看不卡| 国内精品久久久久影院一蜜桃| 久久精品成人欧美大片| 日韩va亚洲va欧美va久久| 久久精品中文无码资源站| 久久精品国产2020| 综合久久精品色| 亚洲国产精品狼友中文久久久 | 精品久久久无码中文字幕天天| 99久久精品国内| 久久精品www人人爽人人| 久久男人Av资源网站无码软件 | 日韩精品久久无码人妻中文字幕 | 综合人妻久久一区二区精品 | 久久精品国产亚洲一区二区三区 | 91久久精品国产免费直播| 国产精品久久久久影视不卡| 2020久久精品国产免费| 77777亚洲午夜久久多喷| 91精品国产色综久久|