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

曠野的呼聲

路漫漫其修遠兮 吾將上下而求索

常用鏈接

統計

最新評論

【原創】boost.compressed_pair源碼剖析

boost.compressed_pair源碼剖析

意義

compressed_pair的某一個模板參數為一個空類的時候將對其進行“空基類優化”,這樣可以使得compressed_pair占用的空間比std::pair的更小。

參考如下代碼:

#include <iostream>

using namespace std;

 

#include <boost/compressed_pair.hpp>

 

class A{};

class B{};

 

int main()

{

   cout<< sizeof( std::pair< A, B > ) << endl;

 

   cout<< sizeof( boost::compressed_pair< A, B > ) << endl;

 

   return 0;

}

在我的機器上,VC2008 SP1輸出21

 

std::pair參考

std::pair被實現為一個結構體,其中VC2008 SP1pair被實現為如下:

template<class _Ty1, class _Ty2>

   struct pair

   {  // store a pair of values

   typedef pair<_Ty1, _Ty2> _Myt;

   typedef _Ty1 first_type;

   typedef _Ty2 second_type;

 

   pair()

      : first(_Ty1()), second(_Ty2())

      {  // construct from defaults

      }

 

   pair(const _Ty1& _Val1, const _Ty2& _Val2)

      : first(_Val1), second(_Val2)

      {  // construct from specified values

      }

 

   template<class _Other1,

      class _Other2>

      pair(const pair<_Other1, _Other2>& _Right)

      : first(_Right.first), second(_Right.second)

      {  // construct from compatible pair

      }

 

   // 刪除了一些不重要的代碼

 

   _Ty1 first; // the first stored value

   _Ty2 second; // the second stored value

};

通過直接訪問firstsecond對數據進行訪問。

 

boost.compressed_pair參考

boost.compressed_pair大概的實現如下:

 

class compressed_pair

{

public:

   typedef T                                                  first_type;

   typedef T                                                  second_type;

   typedef typename call_traits<first_type>::param_type       first_param_type;

   typedef typename call_traits<second_type>::param_type      second_param_type;

   typedef typename call_traits<first_type>::reference        first_reference;

   typedef typename call_traits<second_type>::reference       second_reference;

   typedef typename call_traits<first_type>::const_reference  first_const_reference;

   typedef typename call_traits<second_type>::const_reference second_const_reference;

 

   first_reference       first()       {return base::first();}

   first_const_reference first() const {return base::first();}

 

   second_reference       second()       {return base::second();}

   second_const_reference second() const {return base::second();}

 

   void swap(::boost::compressed_pair<T,T>& y) { base::swap(y); }

};

 

注意這不是完整的代碼,它只是對其實現的一個簡單描述。從中我們可以看出boost.compressed_pair使用成員函數來訪問數據而不是如std::pair一樣直接訪問firstsecond

 

boost.compressed_pair剖析

    boost.compressed_pair的實現依賴于boost.type_traitsboost.call_traitsboost.type_traitsboost提供的一個特征類庫,這是一個強大的庫,可以應用于很多地方。boost的大量組件都依賴于它。boost.call_traits也是一個類似于type_traits的庫,它主要提供的是一些類型調整,通過編譯器演繹我們可以在編譯時得到最好的type,它可以使我們的傳遞的參數等等相關內容總是以最恰當(根據經驗)的方式來進行調用,而且還能在新的C++標準發布之前繞過“引用的引用”問題。

接下來我將剖析支持偏特化版本的compressed_pair的實現,它位于boost\detail\compressed_pair.hpp

 

compressed_pair_switch

這是一個開關工具,用于在后面對各種情況進行開關控制,它的基本實現如下:

template <class T1, class T2, bool IsSame, bool FirstEmpty, bool SecondEmpty>

   struct compressed_pair_switch;

注意,它只是定義而非實現,因此我們無法構造未特化過的compressed_pair_switch。通過查看它的模板參數可以知道后面三個bool代表了三個概念:

l  pair的兩個模板參數是否是相同類型。(去掉cv限定符之后)。

l  第一個模板參數是空的嗎?

l  第二個模板參數是空的嗎?

因此對這三個bool進行有限組合可以得到6種組合,也就出現了接下來我們所看到的6個特化(偏特化)。

template <class T1, class T2>

   struct compressed_pair_switch<T1, T2, false, false, false>

      {static const int value = 0;};

 

   template <class T1, class T2>

   struct compressed_pair_switch<T1, T2, false, true, true>

      {static const int value = 3;};

 

   template <class T1, class T2>

   struct compressed_pair_switch<T1, T2, false, true, false>

      {static const int value = 1;};

 

   template <class T1, class T2>

   struct compressed_pair_switch<T1, T2, false, false, true>

      {static const int value = 2;};

 

   template <class T1, class T2>

   struct compressed_pair_switch<T1, T2, true, true, true>

      {static const int value = 4;};

 

   template <class T1, class T2>

   struct compressed_pair_switch<T1, T2, true, false, false>

      {static const int value = 5;};

現在我們已經偏特化了6個不同的開關,它們將在最終實現compressed_pair的過程中發揮巨大的作用。注意每個類中的value,這個常量值代表了它的版本。

 

compressed_pair_imp

它作為最終compressed_pair的基類存在,它的聲明如下:

template <class T1, class T2, int Version> class compressed_pair_imp;

注意第三個參數Version,在最終的實現中它將被以compressed_pair_switch::value來具現化。

接下來按照compressed_pair_switch6種版本所說明的6中組合情況分別實現其對應的compressed_pair_imp。在文章最開始的時候我們的簡單程序發現std::pair由于直接采用組合T1T2而無法使之成功的應用“空基類優化”,使得其占用空間的大小是2.如果compressed_pair_imp也直接按照這種組合來實現的話,那么所謂“壓縮”便不會有任何意義。所以compressed_pair_imp對應不同組合情況有不同的實現,比如說對于版本1

template <class T1, class T2>

   struct compressed_pair_switch<T1, T2, false, true, false>

      {static const int value = 1;};

這種情況便是指T1T2在去cv限定符之后為不同類型,且第一種類型為空,第二種不為空,那么這時候在實現compressed_pair_imp的時候便取消了T1的數據,源碼如下:

template <class T1, class T2>

   class compressed_pair_imp<T1, T2, 1>

      : protected ::boost::remove_cv<T1>::type

   {

   public:

      typedef T1                                                 first_type;

      typedef T2                                                 second_type;

      typedef typename call_traits<first_type>::param_type       first_param_type;

      typedef typename call_traits<second_type>::param_type      second_param_type;

      typedef typename call_traits<first_type>::reference        first_reference;

      typedef typename call_traits<second_type>::reference       second_reference;

      typedef typename call_traits<first_type>::const_reference  first_const_reference;

      typedef typename call_traits<second_type>::const_reference second_const_reference;

 

      compressed_pair_imp() {}

 

      compressed_pair_imp(first_param_type x, second_param_type y)

         : first_type(x), second_(y) {}

 

      compressed_pair_imp(first_param_type x)

         : first_type(x) {}

 

      compressed_pair_imp(second_param_type y)

         : second_(y) {}

 

      first_reference       first()       {return *this;}

      first_const_reference first() const {return *this;}

 

      second_reference       second()       {return second_;}

      second_const_reference second() const {return second_;}

   private:

      second_type second_;

};

 

      現在回過頭去,對于版本0

template <class T1, class T2>

   struct compressed_pair_switch<T1, T2, false, false, false>

      {static const int value = 0;};

     由于T1t2為不同類型,同時都不為空,因此這種情況下compressed_pairstd::pair是一樣的。

 

     對于版本2

template <class T1, class T2>

   struct compressed_pair_switch<T1, T2, false, false, true>

      {static const int value = 2;};

      T1T2不相同,且T1不為空,T2為空,那么這和版本1的差別就在于t2的數據成員被取消,T1的數據成員存在。

template <class T1, class T2>

   class compressed_pair_imp<T1, T2, 2>

      : protected ::boost::remove_cv<T2>::type

   {

      first_reference       first()       {return first_;}

      first_const_reference first() const {return first_;}

 

      second_reference       second()       {return *this;}

      second_const_reference second() const {return *this;}

 

   private:

      first_type first_;

   };  // 刪除了某系無關緊要的代碼

     

      版本3

template <class T1, class T2>

   struct compressed_pair_switch<T1, T2, false, true, true>

      {static const int value = 3;};

     T1t2不相同,且兩者均為空。這種時候compressd_pair_imp不再需要任何數據成員,因此其精簡版的定義如下:

template <class T1, class T2>

   class compressed_pair_imp<T1, T2, 3>

      : protected ::boost::remove_cv<T1>::type,

        protected ::boost::remove_cv<T2>::type

   {

      first_reference       first()       {return *this;}

      first_const_reference first() const {return *this;}

 

      second_reference       second()       {return *this;}

      second_const_reference second() const {return *this;}

      //

      // no need to swap empty bases:

      void swap(::boost::compressed_pair<T1,T2>&) {}

   };

     在這里面我們可以看到它的交換動作根本什么也沒做。而且也沒有數據成員,但是其占用空間大小依然是1.

 

版本4定義了T1T2相同,且均為空的特殊情況:

template <class T1, class T2>

   struct compressed_pair_switch<T1, T2, true, true, true>

      {static const int value = 4;};

   template <class T1, class T2>

   class compressed_pair_imp<T1, T2, 4>

      : protected ::boost::remove_cv<T1>::type

   {

      first_reference       first()       {return *this;}

      first_const_reference first() const {return *this;}

 

      second_reference       second()       {return m_second;}

      second_const_reference second() const {return m_second;}

 

      void swap(::boost::compressed_pair<T1,T2>&) {}

   private:

      T2 m_second;

   };

     既然T1T2均為空,那么為何還要保存一個T2的數據呢?這是為了防止first()second()所返回的對象的地址相同,這是很郁悶的一件事情。

版本5定義了T1T2相同,且均不為空的情況:

template <class T1, class T2>

   class compressed_pair_imp<T1, T2, 5>

   {

      first_reference       first()       {return first_;}

      first_const_reference first() const {return first_;}

 

      second_reference       second()       {return second_;}

      second_const_reference second() const {return second_;}

 

      void swap(::boost::compressed_pair<T1, T2>& y)

      {

         cp_swap(first_, y.first());

         cp_swap(second_, y.second());

      }

   private:

      first_type first_;

      second_type second_;

};

     這個版本并沒有什么特殊之處。

 

compressed_pair

     最終的實現通過繼承compressed_pair_imp來實現,而上述的compressed_pair_imp都有一個Version模板參數,通過編譯時推斷出的compressed_pair_switch的數據value則可得到與其對應的基類。

template <class T1, class T2>

class compressed_pair

   : private ::boost::details::compressed_pair_imp<T1, T2,

             ::boost::details::compressed_pair_switch<

                    T1,

                    T2,

                    ::boost::is_same<typename remove_cv<T1>::type, typename remove_cv<T2>::type>::value,

                    ::boost::is_empty<T1>::value,

                    ::boost::is_empty<T2>::value>::value>

 

template <class T>

class compressed_pair<T, T>

   : private details::compressed_pair_imp<T, T,

             ::boost::details::compressed_pair_switch<

                    T,

                    T,

                    ::boost::is_same<typename remove_cv<T>::type, typename remove_cv<T>::type>::value,

                    ::boost::is_empty<T>::value,

                    ::boost::is_empty<T>::value>::value>

 

      這是按照T1T2是否相同所不同的實現。它們的區別主要在構造函數的實現上:

explicit compressed_pair(first_param_type x) : base(x) {}

    explicit compressed_pair(second_param_type y) : base(y) {}

    對于T1T2相同的情況,上面的這段代碼不是合法的重載。

 

    ::boost::is_same將對T1T2cv限定符之后的類型進行比較,結果value是一個bool值常量,對應于compressed_pair_switch第三個模板參數。::boost::is_empty可以判斷類型是否是空類型,其值value也是一個bool值常量。這樣便可推斷出該繼承哪一個版本的compressed_pair_imp

 

其它重點

     這里將繼續探討一些有意義的技巧。

 

繼承

     在這里我們將聚焦compressed_pair_imp的實現,其中除了第0版本和第5版本之外,其它的實現均不同程度的使用了繼承,比如第1版:

template <class T1, class T2>

   class compressed_pair_imp<T1, T2, 1>

      : protected ::boost::remove_cv<T1>::type

這個版本是:

template <class T1, class T2>

   struct compressed_pair_switch<T1, T2, false, true, false>

      {static const int value = 1;};

T1T2不同,且只有T1為空,那么它繼承去cv限定符之后的T1類型。它的意義何在呢?

參考第1版本的compressed_pair_imp的實現可以看到:

first_reference       first()       {return *this;}

      first_const_reference first() const {return *this;}

當需求要T1的類型對象的時候,直接返回的是compressed_pair_imp對象自己,如果不繼承的話這個動作將是非法的。這就是為什么有這樣一個繼承的原因。

對于版本3

template <class T1, class T2>

   class compressed_pair_imp<T1, T2, 3>

      : protected ::boost::remove_cv<T1>::type,

        protected ::boost::remove_cv<T2>::type

還可以看到:

first_reference       first()       {return *this;}

      first_const_reference first() const {return *this;}

 

      second_reference       second()       {return *this;}

      second_const_reference second() const {return *this;}

實際上道理是一樣的。同時它還解決了構造函數中的這個問題:

compressed_pair_imp(first_param_type x, second_param_type y)

         : first_type(x), second_type(y) {}

因為first_typesecond_type為其基類,這樣的調用才合法。

還有一個猜想就是如果T1是內置類型的話,比如說int,那么繼承int會是合法的代碼嗎?實際上我們并不需要擔心這個,因為int不會通過::boost::is_empty測試。boost::is_empty<int>::value將得到false,因此不會被編譯器演繹到這一步。

 

 

        [本文原創]相關連接:
1. 本文的Word工整版:http://ishare.iask.sina.com.cn/f/5144299.html
2. 關于空基類優化:http://user.qzone.qq.com/84638372/blog/1242484131

posted on 2009-05-18 11:09 董波 閱讀(1247) 評論(0)  編輯 收藏 引用


只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲伦理久久| 国产欧美日韩综合| 欧美 日韩 国产一区二区在线视频| 欧美伊人影院| 欧美a级一区| 国产精品欧美一区二区三区奶水 | 亚洲欧美日韩中文播放| 久久成人综合网| 亚洲日本va午夜在线影院| 中文国产成人精品久久一| 久久精品1区| 久久综合伊人77777尤物| 国产精品欧美日韩久久| 亚洲精品1区| 久久久久一区二区| 欧美国产欧美亚洲国产日韩mv天天看完整| 欧美日韩国产免费观看| 韩国成人理伦片免费播放| 亚洲深夜福利| 亚洲国产高清一区二区三区| 性亚洲最疯狂xxxx高清| 欧美日韩国产精品专区| 亚洲夫妻自拍| 蜜臀久久久99精品久久久久久| 亚洲亚洲精品三区日韩精品在线视频| 免费在线看一区| 国内精品久久久久久影视8| 亚洲欧美另类在线观看| 亚洲久久成人| 久久久久久久97| 国产一区在线看| 久久久久久久综合| 欧美亚洲视频在线看网址| 国产精品日韩专区| 香蕉亚洲视频| 亚洲一二三区视频在线观看| 欧美日韩国产一区二区| 精品成人一区| 久久青草福利网站| 久久er精品视频| 狠狠色综合一区二区| 久久阴道视频| 美女黄毛**国产精品啪啪| 一区二区三区亚洲| 久久成人免费网| 午夜在线成人av| 国内精品一区二区三区| 久久一区国产| 久久久久久久综合色一本| 国产欧美日韩综合一区在线播放| 性做久久久久久| 午夜亚洲精品| 影音先锋国产精品| 亚洲第一网站免费视频| 欧美精品在线观看91| 91久久一区二区| 欧美亚男人的天堂| 国产欧美一区二区三区在线老狼| 亚洲国产你懂的| 欧美不卡视频一区| 裸体丰满少妇做受久久99精品| 悠悠资源网久久精品| 欧美大香线蕉线伊人久久国产精品| 亚洲欧美成人在线| 国产午夜精品在线观看| 一本久久综合亚洲鲁鲁| 99精品国产福利在线观看免费| 国产精品久久999| 久久久国产成人精品| 久久久成人精品| 亚洲精品国产精品国自产在线 | 亚洲欧美中文在线视频| 国产一区免费视频| 欧美国产大片| 欧美日韩不卡视频| 久久激情五月激情| 欧美精品一区二区三区一线天视频| 亚洲尤物视频网| 久久精品日韩一区二区三区| 最新中文字幕亚洲| 亚洲人成网站影音先锋播放| 国产精品久久久久久久久久直播| 久久久亚洲综合| 欧美色123| 亚洲大片在线观看| 国产一区二区三区四区老人| 久久国产主播精品| 欧美色欧美亚洲另类二区| 美日韩精品视频| 国产精品家庭影院| 亚洲国产色一区| 国产婷婷一区二区| 久久国产精品毛片| 欧美色视频在线| 欧美国产日本高清在线| 国产色爱av资源综合区| 亚洲久久成人| 国产美女诱惑一区二区| 99视频一区二区三区| 亚洲精品国产无天堂网2021| 亚洲欧美日韩国产综合| 亚洲无线视频| 欧美日韩成人在线视频| 久久精彩视频| 国产精品人成在线观看免费| 亚洲精品精选| 亚洲精品久久久久中文字幕欢迎你 | 欧美三级日韩三级国产三级| 久久久成人精品| 欧美日韩在线播放三区| 亚洲激情黄色| 亚洲精品一区二区三区福利| 久久超碰97人人做人人爱| 99视频精品在线| 久久精品国产亚洲aⅴ| 久久人人九九| 久久综合九九| 国产精品久久一卡二卡| 亚洲精选久久| 亚洲一区二区三区精品在线| 欧美日韩成人一区| 亚洲伦理久久| 亚洲一区二区三区免费观看| 国产精品理论片在线观看| 亚洲女人天堂av| 久久久国产精品一区二区三区| 国产在线不卡| 欧美成人免费全部| 一二三区精品| 久久―日本道色综合久久| 亚洲二区在线视频| 欧美成人在线网站| 亚洲深夜福利| 欧美69wwwcom| 亚洲一级影院| 黄色成人av在线| 欧美日本韩国一区| 欧美一区二区日韩一区二区| 欧美激情亚洲一区| 亚洲欧美三级伦理| 尤物yw午夜国产精品视频| 欧美大片第1页| 亚洲欧美日韩在线播放| 欧美激情bt| 欧美一区2区三区4区公司二百| 亚洲第一在线综合网站| 国产精品成人一区| 久久综合亚州| 欧美夜福利tv在线| 亚洲精品欧美专区| 久久尤物电影视频在线观看| 一本色道久久综合亚洲二区三区| 国产一区二区久久久| 欧美日韩国产欧美日美国产精品| 久久精品91久久香蕉加勒比| 亚洲最快最全在线视频| 免费在线欧美视频| 久久岛国电影| 亚洲男女毛片无遮挡| 亚洲精品久久久久中文字幕欢迎你 | 在线观看成人av| 国产精品免费观看在线| 欧美1级日本1级| 久久精品99国产精品| 亚洲视频一区二区在线观看 | 国产精品午夜视频| 欧美片网站免费| 免费日韩视频| 久热成人在线视频| 久久精品导航| 欧美一区二区三区四区在线观看 | 亚洲理伦电影| 亚洲高清不卡| 伊人夜夜躁av伊人久久| 国产网站欧美日韩免费精品在线观看| 欧美色道久久88综合亚洲精品| 欧美国产日产韩国视频| 欧美成人免费全部| 欧美激情久久久| 欧美国产欧美亚州国产日韩mv天天看完整| 久久久精品免费视频| 久久精品99无色码中文字幕| 欧美在线www| 欧美一区二区精品| 亚洲一区二区三区四区视频| 亚洲一区二区三区视频播放| 亚洲视屏在线播放| 亚洲男女自偷自拍| 午夜精品在线视频| 欧美一区二区三区另类| 久久国产精品一区二区| 久久久精彩视频| 麻豆国产精品va在线观看不卡 | 一区二区视频在线观看| 在线观看日韩av| 亚洲国产一区在线| 夜久久久久久| 亚洲欧美日韩人成在线播放| 午夜精品久久久久久久久久久久久 | 欧美在线网址|