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

woaidongmao

文章均收錄自他人博客,但不喜標題前加-[轉貼],因其丑陋,見諒!~
隨筆 - 1469, 文章 - 0, 評論 - 661, 引用 - 0
數據加載中……

Traits技術

.概念

Trait的中文意思就是特性,Traits就像特性萃取機,榨取不同類的特性,以便能統一處理。Traits依靠顯式模板特殊化(explicit template specialization)來把代碼中因類型不同而發生變化的片斷拖出來,用統一的接口來包裝。這個接口可以包含一個C++類所能包含的任何東西:內嵌類型,成員函數,成員變量,作為客戶的模板代碼可以通過traits模板類所公開的接口來間接訪問之。花了點時間,憋了一個示例程序,雖然沒什么實際意義,但對于理解traits來說,可能會有幫助。

#include <iostream>

class CM {};

class CA {};

class CIM {};

template< typename T >

class Traits

{

public:

    typedef unsigned int ValueType;

    Traits() : m_rate( 2 ) {}

    ValueType ComputeValue( ValueType value ) { return value * m_rate; }

private:

    ValueType m_rate;

};

template<>

class Traits< CM >

{

public:

    typedef double ValueType;

    Traits() : m_rate( 1.2 ) {}

    ValueType ComputeValue( ValueType value ) { return value * m_rate; }

private:

    ValueType m_rate;

};

template<>

class Traits< CA >

{

public:

    typedef int ValueType;

    Traits() : m_rate( -1 ) {}

    ValueType ComputeValue( ValueType value ) { return value * m_rate; }

private:

    ValueType m_rate;

};

template< typename T >

struct ValueCount

{

    void result()

    {

        Traits< T > traits;

        Traits< T >::ValueType value = static_cast< Traits< T >::ValueType >( 2 );

        std::cout << traits.ComputeValue( value ) << std::endl;

    }

};

int main()

{

    ValueCount< CIM > cim;

    cim.result();

    ValueCount< CM > cm;

    cm.result();

    ValueCount< CA > ca;

    ca.result();

    return 0;

}

代碼中的Tratis類對于CACM有顯式的特化實現,其中ValueType類型和rate與默認的實現不同,在類ValueCount中,利用Tratis對不同類型用一個統一的接口符號處理。程序的運行結果是42.4-2。(這是我的第一個模版程序J

2.SGI STL中的__type_traits

SGI實現版的STL中,為了獲取高效率,提供了__type_traits,用來提取類的信息,比如類是否擁有trival的構造、析構、拷貝、賦值操作,然后跟據具體的信息,就可提供最有效率的操作。以下摘錄cygwingcc3.3源碼,有改動,在<type_traits.h>中。

struct __true_type {};

struct __false_type {};

template <class _Tp>

struct __type_traits {

    typedef __false_type has_trivial_default_constructor;

    typedef __false_type has_trivial_copy_constructor;

    typedef __false_type has_trivial_assignment_operator;

    typedef __false_type has_trivial_destructor;

    typedef __false_type is_POD_type;

};

對于普通類來講,為了安全起見,都認為它們擁有non-trival的構造、析構、拷貝、賦值函數,POD是指plain old data。接下來對C++的原生類型(boolint double之類)定義了顯式的特化實現,以double為例:

template<> struct __type_traits<long double> {

    typedef __true_type has_trivial_default_constructor;

    typedef __true_type has_trivial_copy_constructor;

    typedef __true_type has_trivial_assignment_operator;

    typedef __true_type has_trivial_destructor;

    typedef __true_type is_POD_type;

};

還有,對所有的原生指針來講,它們的構造、析構等操作也是trival的,因此有:

template <class _Tp>

struct __type_traits<_Tp*> {

    typedef __true_type has_trivial_default_constructor;

    typedef __true_type has_trivial_copy_constructor;

    typedef __true_type has_trivial_assignment_operator;

    typedef __true_type has_trivial_destructor;

    typedef __true_type is_POD_type;

};

簡化<stl_algobase.h>copy的部分代碼來說明對__type_traits的應用。

template<typename _Tp>

inline _Tp*

__copy_trivial(const _Tp* __first, const _Tp* __last, _Tp* __result)

{

    memmove(__result, __first, sizeof(_Tp) * (__last - __first));

    return __result + (__last - __first);

}

template<typename _Tp>

inline _Tp*

__copy_aux (_Tp* __first, _Tp* __last, _Tp* __result, __true_type)

{ return __copy_trivial(__first, __last, __result); }

template<typename _Tp>

inline _Tp*

__copy_aux (_Tp* __first, _Tp* __last, _Tp* __result, __false_type)

{ 另外處理; }

template<typename _InputIter, typename _OutputIter>

inline _OutputIter

copy (_InputIter __first, _InputIter __last, _OutputIter __result)

{

    typedef typename iterator_traits<_InputIter>::value_type _ValueType;

    typedef typename __type_traits<_ValueType>::has_trivial_assignment_operator _Trivial;

    return __copy_aux(__first, __last, __result, _Trivial());

}

Copy函數利用__type_traits判斷當前的value_type是否有trival的賦值操作,如果是,則產生類__true_type的實例,編譯時選擇__copy_trivial函數進行memmove,效率最高。如果是non-trival的賦值操作,則另作處理,效率自然低些。__true_type__false_type之所以是類,就因為C++的函數重載是根據類型信息來的,不能依據參數值來判別。使用SGI STL時,可以為自己的類定義__type_traits顯式特化版本,以求達到高效率。

3. STL中的iterator_traits

iterator_traits提供5個特性的提取,至于為什么是5個以及iterator的分類,以后再議。代碼包含在<stl_iterator_base_types.h>中。

template<typename _Category, typename _Tp, typename _Distance = ptrdiff_t,

typename _Pointer = _Tp*, typename _Reference = _Tp&>

struct iterator

{

    typedef _Category iterator_category;

    typedef _Tp value_type;

    typedef _Distance difference_type;

    typedef _Pointer pointer;

    typedef _Reference reference;

};

template<typename _Iterator>

struct iterator_traits {

    typedef typename _Iterator::iterator_category iterator_category;

    typedef typename _Iterator::value_type value_type;

    typedef typename _Iterator::difference_type difference_type;

    typedef typename _Iterator::pointer pointer;

    typedef typename _Iterator::reference reference;

};

原生指針(如int*double*)也是iterator,但它不是類,無法提取出value_type,所以要對原生指針和const原生指針進行顯式特化。

template<typename _Tp>

struct iterator_traits<_Tp*> {

    typedef random_access_iterator_tag iterator_category;

    typedef _Tp value_type;

    typedef ptrdiff_t difference_type;

    typedef _Tp* pointer;

    typedef _Tp& reference;

};

template<typename _Tp>

struct iterator_traits<const _Tp*> {

    typedef random_access_iterator_tag iterator_category;

    typedef _Tp value_type;

    typedef ptrdiff_t difference_type;

    typedef const _Tp* pointer;

    typedef const _Tp& reference;

};

現在,對于所有的iterator都可以正確的提取出以上5個特性。

下面解釋iterator_categoryiterator共分為5類,input_iteratoroutput_iteratorforward_iteratorbidirectional_iteratorrandom_access_iterator。其中forward_iteratorinput_iteratoroutput_iterator的強化(refinement),bidirectional_iteratorforward_iterator的強化,random_access_iteratorbidirectional_iterator的強化。由于5iterator的性質的同異,需要對它們的種類進行區分,制定特化的函數以達到最優的效率,就像上一節的type_traits一樣。需要強調的是,強化不是繼承,C++重載機制支持對繼承類的正確選擇。由于不同iterator有共同的操作,在iterator_category中建立繼承關系可以簡化大部分特化函數的實現。

struct input_iterator_tag {};

struct output_iterator_tag {};

struct forward_iterator_tag : public input_iterator_tag {};

struct bidirectional_iterator_tag : public forward_iterator_tag {};

struct random_access_iterator_tag : public bidirectional_iterator_tag {};

iterator_category帶來的效率優化,可由在<stl_iterator_base_funcs.h>內的兩個函數看出,一個是用來計算兩個iterator的距離__distance,一個是將iterator累進n次的__advance。列舉__distance的代碼如下,有改動。

template<typename _InputIterator>

inline typename iterator_traits<_InputIterator>::difference_type

__distance(_InputIterator __first, _InputIterator __last,

input_iterator_tag)

{

    typename iterator_traits<_InputIterator>::difference_type __n = 0;

    while (__first != __last) {

        ++__first; ++__n;

    }

    return __n;

}

template<typename _RandomAccessIterator>

inline typename iterator_traits<_RandomAccessIterator>::difference_type

__distance(_RandomAccessIterator __first, _RandomAccessIterator __last,

random_access_iterator_tag)

{ return __last - __first; }

template<typename _Iter>

inline typename iterator_traits<_Iter>::iterator_category

__iterator_category(const _Iter&)

{ return typename iterator_traits<_Iter>::iterator_category(); }

template<typename _InputIterator>

inline typename iterator_traits<_InputIterator>::difference_type

distance(_InputIterator __first, _InputIterator __last)

{ return __distance(__first, __last, __iterator_category(__first)); }

4. traits技術還有很夸張的應用。《C++ 設計新思維:范型編程與設計模式之應用》中有體現,或者,泛型編程還得依賴traits技術,也許以后的C++會從語言特性上支持traits

 

posted on 2008-11-08 22:48 肥仔 閱讀(2366) 評論(0)  編輯 收藏 引用 所屬分類: C++ 模板

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲第一精品久久忘忧草社区| 久久www成人_看片免费不卡| 先锋影音久久久| 欧美激情精品久久久久久免费印度| 久久国产精品高清| 国产精品久久久久7777婷婷| 99在线|亚洲一区二区| 亚洲午夜激情| 黄色成人在线| 一色屋精品视频在线观看网站| 欧美一级电影久久| 性做久久久久久免费观看欧美 | 在线观看国产成人av片| 亚洲欧美日韩在线高清直播| 久久伊人一区二区| 亚洲娇小video精品| 日韩视频中文| 欧美jizz19hd性欧美| 亚洲精品一级| 欧美专区亚洲专区| 亚洲国产小视频在线观看| 欧美日韩精品三区| 亚洲欧美精品| 亚洲电影天堂av| 亚洲午夜在线观看视频在线| 国产一区自拍视频| 欧美日韩精品一区视频| 久久久xxx| 亚洲一区在线观看免费观看电影高清| 午夜久久福利| 一区二区三区精品| 激情综合自拍| 欧美三区视频| 免费观看成人| 亚洲综合首页| 亚洲看片网站| 蜜臀av在线播放一区二区三区 | 亚洲高清不卡| 国产午夜精品久久久久久免费视| 欧美成人精品一区| 欧美亚洲在线视频| 在线一区二区三区做爰视频网站| 久久综合伊人77777麻豆| 亚洲一区二区精品| 日韩一区二区福利| 亚洲国产色一区| 精品69视频一区二区三区| 国产精品theporn88| 欧美黄色一级视频| 免费在线欧美黄色| 欧美影院午夜播放| 性高湖久久久久久久久| 中文国产成人精品| 一本久久a久久免费精品不卡| 欧美国产精品专区| 欧美成人久久| 亚洲第一区色| 亚洲电影免费观看高清完整版在线观看 | 久久精品中文字幕一区| 在线视频免费在线观看一区二区| 欧美高清视频一区二区| 久久久噜噜噜久久中文字免| 午夜一区二区三区在线观看| 亚洲一区二区三区视频| 亚洲视频播放| 亚洲视频欧美在线| 亚洲免费成人| 日韩一级黄色片| 亚洲一二三四久久| 亚洲小说欧美另类社区| 午夜精品一区二区在线观看| 久久精品91| 六月婷婷久久| 亚洲高清视频在线观看| 最新国产成人在线观看| 亚洲精品乱码视频 | 先锋影音久久| 久久噜噜亚洲综合| 欧美激情麻豆| 欧美日韩视频在线第一区| 欧美日韩在线精品一区二区三区| 欧美午夜影院| 韩国一区电影| 亚洲人成人一区二区在线观看| 亚洲精品视频一区二区三区| 这里只有精品在线播放| 欧美一区在线直播| 久久女同互慰一区二区三区| 美日韩精品免费| 亚洲伦理自拍| 久久久久久久97| 欧美色图麻豆| 1024国产精品| 午夜精品偷拍| 女人天堂亚洲aⅴ在线观看| 亚洲人成毛片在线播放| 午夜精品久久久久| 欧美黑人在线播放| 激情成人综合| 亚洲影视在线| 欧美高清视频在线观看| 亚洲免费大片| 久久综合一区| 国产老女人精品毛片久久| 亚洲国产日韩一区二区| 久久精品亚洲一区| 99热在线精品观看| 久久久免费精品| 国产美女精品视频| 一个色综合导航| 美腿丝袜亚洲色图| 午夜精品视频在线观看一区二区| 欧美精品18videos性欧美| 日韩视频一区二区| 亚洲午夜一二三区视频| 欧美ed2k| 欧美一二三视频| 国产精品av一区二区| 亚洲欧洲在线播放| 久久综合电影一区| 亚洲综合国产| 欧美午夜三级| 在线亚洲观看| 亚洲人午夜精品免费| 另类图片国产| 亚洲国产精品久久久久秋霞不卡| 久久精品亚洲精品| 性欧美办公室18xxxxhd| 欧美性色综合| 亚洲影院污污.| 99国产精品国产精品毛片| 欧美国产国产综合| 亚洲精品视频免费观看| 蜜桃av一区二区三区| 老司机成人在线视频| 黄色资源网久久资源365| 久久综合久久综合久久| 久久精品系列| 亚洲国产精品久久久| 亚洲国语精品自产拍在线观看| 欧美大片一区| 亚洲午夜伦理| 亚洲综合成人在线| 国产午夜一区二区三区| 久久中文字幕一区二区三区| 久久精品人人爽| 一区二区三区在线观看视频 | 99re热这里只有精品免费视频| 亚洲国产视频a| 欧美一区二区三区精品| 在线视频免费在线观看一区二区| 国产精品乱子乱xxxx| 久久xxxx精品视频| 久久青草欧美一区二区三区| 91久久中文字幕| 亚洲视频自拍偷拍| 国产一区欧美日韩| 欧美大片在线看| 欧美日韩国产美女| 久久精品在这里| 久久亚洲精品一区二区| 亚洲欧洲精品一区| a91a精品视频在线观看| 国产亚洲欧洲一区高清在线观看 | 欧美视频网站| 欧美一区二区三区免费视频 | 国产欧美1区2区3区| 久久亚洲高清| 欧美日本在线视频| 一区二区欧美日韩视频| 亚洲嫩草精品久久| 亚洲激情图片小说视频| 亚洲国产精品精华液2区45| 国产欧美日韩三区| 国产精品揄拍500视频| 国产精品美女久久久久av超清| 久久成人这里只有精品| 久久久亚洲一区| 欧美日韩一区二区三区高清| 一区二区在线不卡| 亚洲电影免费在线观看| 日韩一区二区精品视频| 亚洲午夜未删减在线观看| 欧美成人免费va影院高清| 午夜在线一区二区| 国产精品久久久久久久电影| 永久域名在线精品| 久久视频精品在线| 亚洲国产另类久久精品| 欧美激情精品久久久久久免费印度| 中文日韩在线| 欧美精品一区二区视频| 亚洲电影在线看| 久久精品99无色码中文字幕| 欧美中文在线视频| 亚洲国产精品黑人久久久| 欧美大色视频| 欧美视频在线视频| 亚洲视频免费观看| 欧美国产精品中文字幕|