3.泛型編程

項目地址

概觀(Overview)

  • 泛型編程是一種編程方式,這種方法將型別(type)以一種<code>to-be-specified-later</code>的方式給出闻蛀,等到需要調用的時候,再以參數(shù)方式考榨,通過具體的崩掘、特定的型別實例化(instantiate)一個具體的方法或對象
  • 泛型編程作為一種編程的想法或思維,不依賴于具體的語言
  • 大多數(shù)面向對象的語言(OO languages)都支持泛型編程惩激,比如:C++,C#,Java,Ada,...
  • C++里面的泛型是通過模板以及相關性質表現(xiàn)的

關聯(lián)特性(Traits)

特性(1)

  • 什么是traits以及為什么使用traits?
  • 假設給定一個數(shù)組店煞,計算數(shù)組中所有元素的和:
    A[0] A[1] ... A[n] sum(A)
  • 我們可以很直接的寫出如下的計算函數(shù):

template <typename T> inline T Sigma(const T const* start, const T const* end) {
    T total = T();  //suppose T() actually creates a zero value
    while (start != end) {
        total += *start++;
    }

    return total;
}

特性(2)

  • 什么是traits以及為什么使用traits?(續(xù))
  • 當我們使用char型別調用模板函數(shù)時蟹演,問題來了:

char szNames[] = "abd";
std::size_t nLength = strlen(szNames);
char* p = szNames;
char* q = szNames + nLength;
printf("Sigma(szNames) = %d\n", Sigma(p,q));

traits2
  • 調用Sigma(szNames)的結果是38(=0x26)!而并非期盼的值(97+98+99=294)
  • 原因是顯而易見的:char型別無法存下294這個值风钻!
  • 如果要得到正確的結果,我們不得不強制使用int型別:
    int s = Sigma<int>(p,q);
  • 但是這種不必要的轉換是完全可以避免的酒请!

特性(3)

  • 什么是traits以及為什么使用traits?(續(xù))
  • 解決的方法是:為每個Sigma函數(shù)的參數(shù)型別T創(chuàng)建一種關聯(lián)(association)骡技,關聯(lián)的型別就是用來存儲Sigma結果的型別
  • 這種關聯(lián)可以看作是型別T的一種特性(characteristic of the type T),因此Sigma函數(shù)返回值的型別叫做T的trait
  • T與其trait的關系推演如下:
    T -> association -> characteristic of T -> another type -> trait!
  • Traits可以實現(xiàn)為模板類羞反,而關聯(lián)(association)則是針對每個具體型別T的特化布朦。在這個例子里我們將traits命名為SigmaTraits,叫做(traits template)

特性(4)

  • Traits實現(xiàn)

template <typename T> class SigmaTraits{};
template <> class SigmaTraits<char> {
    public: typedef int ReturnType;
};

template <> class SigmaTraits<short> {
    public: typedef int ReturnType;
};

template <> class SigmaTraits<int> {
    public: typedef long ReturnType;
};

template <> class SigmaTraits<unsigned int> {
    public: typedef unsigned long ReturnType;
};

template <> class SigmaTraits<float> {
    public: typedef double ReturnType;
};

traits4

特性(5)

  • Traits實現(xiàn)(續(xù))
  • 模板類SigmaTraits叫做traits template昼窗,它含有其參數(shù)型別T的一個特性(trait)是趴,即ReturnType
  • 現(xiàn)在Sigma函數(shù)可以改寫如下:

template <typename T>
inline typename SigmaTraits<T>::ReturnType Sigma(const T const* start, const T const* end)
{
    typedef typename SigmaTraits<T>::ReturnType ReturnType;
    ReturnType s = ReturnType();
    while (start != end)
        s += *start++;

    return s;
}

特性(6)

  • Traits實現(xiàn)
  • 現(xiàn)在如果我們以char為型別調用Sigma將得到預想中的結果:

char szNames[] = "abc";
std::size_t nLength = strlen(szNames);
char* p = szNames;
char* q = szNames + nLength;
printf("Sigma(szNames) = %d\n", Sigma(p,q));

  • 雖然傳入參數(shù)T的型別是char,但是返回的型別卻是int澄惊,原因就在于template <> class SigmaTraits<char>的返回值變成了int(通過typedef int ReturnType)

迭代器(1)

  • 什么是迭代器唆途?

  • 迭代器是指針的泛化(generalization of pointers)

  • 迭代器本身是一個對象,指向另外一個(可以被迭代的)對象

  • 用來迭代一組對象掸驱,即如果迭代器指向一組對象中的某個元素肛搬,則通過increment以后它就可以指向這組對象中的下一個元素

  • 在STL中迭代器是容器與算法之間的接口

  • 算法通常以迭代器作為輸入參數(shù)

  • 容器只要提供一種方式,可以讓迭代器訪問容器中的元素即可

迭代器(2)

  • 迭代器的基本思想
  • 在STL中毕贼,迭代器最重要的思想就是分離算法和容器温赔,使之不需要互相依賴
  • 迭代器將算法和容器粘合(stick)在一起從而使得一種算法的實現(xiàn)可以運用到多種不同的容器上,如下面的例子所示鬼癣,find算法接受一對迭代器陶贼,分別指向容器的開始位置和最終位置:

template<class _Init, class _Ty>
inline _Init find(_Init _First, _Init _Last, const _Ty& _Val) {
    //find first matching _Val
    for (; _First != _Last; ++_First)
        if (*_First == _Val)
            break;
    return (_First);
}

迭代器(3)

  • 迭代器的基本思想(續(xù))
  • find算法對于不同的容器,比如vector,list,deque均適用:

std::vector<int> v(...);
std::list<int> l(...);
std::deque<int> d(...);

std::vector<int>::iterator itv = std::find(v.begin(), v.end(), elementToFind)
std::list<int>::iterator itl = std::find(l.begin(), l.end(), elementToFind)
std::deque<int>::iterator itd = std::find(d.begin(), d.end(), elementToFind)

  • 每種容器都有其對應的迭代器
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末待秃,一起剝皮案震驚了整個濱河市骇窍,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌锥余,老刑警劉巖腹纳,帶你破解...
    沈念sama閱讀 219,539評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡嘲恍,警方通過查閱死者的電腦和手機足画,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,594評論 3 396
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來佃牛,“玉大人淹辞,你說我怎么就攤上這事》溃” “怎么了象缀?”我有些...
    開封第一講書人閱讀 165,871評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長爷速。 經常有香客問我央星,道長,這世上最難降的妖魔是什么惫东? 我笑而不...
    開封第一講書人閱讀 58,963評論 1 295
  • 正文 為了忘掉前任莉给,我火速辦了婚禮,結果婚禮上廉沮,老公的妹妹穿的比我還像新娘颓遏。我一直安慰自己,他們只是感情好滞时,可當我...
    茶點故事閱讀 67,984評論 6 393
  • 文/花漫 我一把揭開白布叁幢。 她就那樣靜靜地躺著,像睡著了一般坪稽。 火紅的嫁衣襯著肌膚如雪曼玩。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,763評論 1 307
  • 那天刽漂,我揣著相機與錄音演训,去河邊找鬼。 笑死贝咙,一個胖子當著我的面吹牛样悟,可吹牛的內容都是我干的。 我是一名探鬼主播庭猩,決...
    沈念sama閱讀 40,468評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼氯庆,長吁一口氣:“原來是場噩夢啊……” “哼勇劣!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 39,357評論 0 276
  • 序言:老撾萬榮一對情侶失蹤翁巍,失蹤者是張志新(化名)和其女友劉穎永罚,沒想到半個月后勘纯,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經...
    沈念sama閱讀 45,850評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡论咏,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 38,002評論 3 338
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了颁井。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片厅贪。...
    茶點故事閱讀 40,144評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖雅宾,靈堂內的尸體忽然破棺而出养涮,到底是詐尸還是另有隱情,我是刑警寧澤眉抬,帶...
    沈念sama閱讀 35,823評論 5 346
  • 正文 年R本政府宣布贯吓,位于F島的核電站,受9級特大地震影響蜀变,放射性物質發(fā)生泄漏悄谐。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,483評論 3 331
  • 文/蒙蒙 一昏苏、第九天 我趴在偏房一處隱蔽的房頂上張望尊沸。 院中可真熱鬧威沫,春花似錦贤惯、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,026評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至烟很,卻和暖如春颈墅,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背雾袱。 一陣腳步聲響...
    開封第一講書人閱讀 33,150評論 1 272
  • 我被黑心中介騙來泰國打工恤筛, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人芹橡。 一個月前我還...
    沈念sama閱讀 48,415評論 3 373
  • 正文 我出身青樓毒坛,卻偏偏與公主長得像,于是被迫代替她去往敵國和親林说。 傳聞我的和親對象是個殘疾皇子煎殷,可洞房花燭夜當晚...
    茶點故事閱讀 45,092評論 2 355

推薦閱讀更多精彩內容