C++泛型與多態(tài)(4): Duck Typing

對(duì)于一門強(qiáng)類型的靜態(tài)語(yǔ)言來(lái)說,要想通過運(yùn)行時(shí)多態(tài)來(lái)隔離變化,多個(gè)實(shí)現(xiàn)類就必須屬于同一類型體系。也就是說,它們必須通過繼承的方式,與同一抽象類型建立is-a關(guān)系。

Duck Typing則是一種基于特征,而不是基于類型的多態(tài)方式。事實(shí)上它仍然關(guān)心is-a,只不過這種is-a關(guān)系是以對(duì)方是否具備它所關(guān)心的特征來(lái)確定的。

James Whitcomb Riley在描述這種is-a的哲學(xué)時(shí),使用了所謂的鴨子測(cè)試Duck Test):

當(dāng)我看到一只鳥走路像鴨子,游泳像鴨子,叫聲像鴨子,那我就把它叫做鴨子。(When I see a bird that walks like a duck and swims like a duck and quacks like a duck, I call that bird a duck.)

鴨子測(cè)試
鴨子測(cè)試

Duck Test基于特征的哲學(xué),給設(shè)計(jì)提供了強(qiáng)大的靈活性。動(dòng)態(tài)面向?qū)ο笳Z(yǔ)言,如PythonRuby等,都遵從了這種哲學(xué)來(lái)實(shí)現(xiàn)運(yùn)行時(shí)多態(tài)。下面給出一個(gè)Python的例子:

class Duck:
    def quack(self):
        print("Quaaaaaack!")
    def feathers(self):
        print("The duck has white and gray feathers.")

class Person:
    def quack(self):
        print("The person imitates a duck.")
    def feathers(self):
        print("The person takes a feather from the ground and shows it.")
    def name(self):
        print("John Smith")

def in_the_forest(duck):
    duck.quack()
    duck.feathers()

def game():
    donald = Duck()
    john = Person()
    in_the_forest(donald)
    in_the_forest(john)

game()

但這并不意味著Duck Typing是動(dòng)態(tài)語(yǔ)言的專利。C++作為一門強(qiáng)類型的靜態(tài)語(yǔ)言,也對(duì)此特性有著強(qiáng)有力的支持。只不過,這種支持不是運(yùn)行時(shí),而是編譯時(shí)。

其實(shí)現(xiàn)的方式為:一個(gè)模板類或模版函數(shù),會(huì)要求其實(shí)例化的類型必須具備某種特征,如某個(gè)函數(shù)簽名,某個(gè)類型定義,某個(gè)成員變量等等。如果特征不具備,編譯器會(huì)報(bào)錯(cuò)。

比如下面一個(gè)模板函數(shù):

template <typename T> 
void f(const T& object) 
{ 
  object.f(0); // 要求類型 T 必須有一個(gè)可讓此語(yǔ)句編譯通過的函數(shù)。
} 

對(duì)于這樣一個(gè)函數(shù),下面的四個(gè)類均可以用來(lái)作為其參數(shù)類型。

struct C1 
{
  void f(int); 
};
 
struct C2 
{ 
  int f(char); 
};
 
struct C3 
{ 
  int f(unsigned short, bool isValid = true); 
}; 
 
struct C4
{
  Foo* f(Object*);
};

一旦上述模板函數(shù)實(shí)現(xiàn)為下面的樣子,則只有C2C3可以和f配合工作。

template <typename T> 
void f(const T& object) 
{ 
  int result = object.f(0); 
  // ... 
} 

通過之前的解釋我們不難發(fā)現(xiàn),Duck Typing要表達(dá)的多態(tài)語(yǔ)義如下圖所示:

DuckTyping的語(yǔ)義
DuckTyping的語(yǔ)義

適配器:類型萃取

Duck Typing需要實(shí)例化的類型具備一致的特征,而模板特化的作用正是為了讓不同類型具有統(tǒng)一的特征(統(tǒng)一的操作界面),所以模板特化可以作為Duck Typing與實(shí)例化類型之間的適配器。這種模板特化手段稱為萃取Traits),其中類型萃取最為常見,畢竟類型是模板元編程的核心元素。

所以,類型萃取首先是一種非侵入性的中間層。否則,這些特征就必須被實(shí)例化類型提供,而就意味著,當(dāng)一個(gè)實(shí)例化類型需要復(fù)用多個(gè)Duck Typing模板時(shí),就需要迎合多種特征,從而讓自己經(jīng)常被修改,并逐漸變得龐大和難以理解。

Type Traits的語(yǔ)義
Type Traits的語(yǔ)義

另外,一個(gè)Duck Typing模板,比如一個(gè)通用算法,需要實(shí)例化類型提供一些特征時(shí),如果一個(gè)類型是類,則是一件很容易的事情,因?yàn)槟憧梢栽谝粋€(gè)類里定義任何需要的特征。但如果一個(gè)基本類型也想復(fù)用此通用算法,由于基本類型無(wú)法靠自己提供算法所需要的特征,就必須借助于類型萃取

結(jié)論

這四篇文章所介紹的,就是C++泛型編程的全部關(guān)鍵知識(shí)。

從中可以看出,泛型是一種多態(tài)技術(shù)。而多態(tài)的核心目的是為了消除重復(fù)隔離變化,提高系統(tǒng)的正交性。因而,泛型編程不僅不應(yīng)該被看做奇技淫巧,而是任何一個(gè)追求高效的C++工程師都應(yīng)該掌握的技術(shù)。

同時(shí),我們也可以看出,相關(guān)的思想在其它范式和語(yǔ)言中(FP,動(dòng)態(tài)語(yǔ)言)也都存在。因而,對(duì)于其它范式和語(yǔ)言的學(xué)習(xí),也會(huì)有助于更加深刻的理解泛型,從而正確的使用范型。

最后給出關(guān)于泛型的缺點(diǎn):

  1. 復(fù)雜模板的代碼非常難以理解;
  2. 編譯器關(guān)于模板的出錯(cuò)信息十分晦澀,尤其當(dāng)模板存在嵌套時(shí);
  3. 模板實(shí)例化會(huì)進(jìn)行代碼生成,重復(fù)信息會(huì)被多次生成,這可能會(huì)造成目標(biāo)代碼膨脹;
  4. 模板的編譯可能非常耗時(shí);
  5. 編譯器對(duì)模板的復(fù)雜性往往會(huì)有自己限制,比如當(dāng)使用遞歸時(shí),當(dāng)遞歸層次太深,編譯器將無(wú)法編譯;
  6. 不同編譯器(包括不同版本)之間對(duì)于模板的支持程度不一,當(dāng)存在移植性需求時(shí),可能出現(xiàn)問題;
  7. 模板具有傳染性,往往一處選擇模板,很多地方也必須跟著使用模板,這會(huì)惡化之前的提到的所有問題。

我對(duì)此的原則是:在使用其它非泛型技術(shù)可以同等解決的前提下,就不會(huì)選擇泛型。

相關(guān)鏈接

深入理解C++泛型(1):基礎(chǔ)篇

深入理解C++泛型(2):模板特化

深入理解C++泛型(3):類模板特化

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,321評(píng)論 6 543
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,559評(píng)論 3 429
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,442評(píng)論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,835評(píng)論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,581評(píng)論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,922評(píng)論 1 328
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,931評(píng)論 3 447
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 43,096評(píng)論 0 290
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,639評(píng)論 1 336
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 41,374評(píng)論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,591評(píng)論 1 374
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,104評(píng)論 5 364
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,789評(píng)論 3 349
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,196評(píng)論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,524評(píng)論 1 295
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 52,322評(píng)論 3 400
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,554評(píng)論 2 379

推薦閱讀更多精彩內(nèi)容