Clean C++: 應(yīng)用「準(zhǔn)關(guān)鍵字」定義抽象接口

使用C++定義純粹的抽象接口類型辞槐,與定義普通的類并無區(qū)別掷漱,只是在形態(tài)上具有特殊的表現(xiàn)形式: 它擁有一個(gè)公開的、空實(shí)現(xiàn)的榄檬、虛擬的析構(gòu)函數(shù)卜范。

struct Rule {
  virtual std::string apply(int n) const = 0;
  virtual ~Rule() {}
}

不幸的是,每當(dāng)定義一個(gè)接口類型時(shí)鹿榜,都要小心翼翼地定義虛擬析構(gòu)函數(shù)海雪,這不僅僅是重復(fù)設(shè)計(jì)的問題;更有甚者舱殿,這樣的慣用法必然存在安全的隱患奥裸,萬一存在一個(gè)程序員在喝酒之后定義給你一個(gè)抽象接口呢?

抽象接口

與其讓程序員存在犯罪的風(fēng)險(xiǎn)沪袭,不如預(yù)防錯(cuò)誤發(fā)生湾宙。既然程序員容易遺忘,那么就自動(dòng)「生成代碼」冈绊,這樣便可高枕無憂了侠鳄。

預(yù)防優(yōu)于治療,將錯(cuò)誤扼殺在源頭死宣。

namespace cub {

namespace details {
template <typename T>
struct Interface {
  virtual ~Interface() {
  }
};
}  // namespace details

#define INTERFACE(type) \
struct type : ::cub::details::Interface<type>

} // namespace cub

有了INTERFACE的宏定義伟恶,定義抽象的接口類型,可謂易如反掌毅该。

INTERFACE(Rule) {
  virtual std::string apply(int n) const = 0;
};

擴(kuò)展接口

但是博秫,應(yīng)用INTERFACE定義多重繼承的接口類型時(shí)潦牛,將面臨困境。因此挡育,需要定義另外的兩個(gè)「準(zhǔn)關(guān)鍵字」罢绽,Java程序員對(duì)這兩個(gè)關(guān)鍵字早已司空見慣了。

#define EXTENDS(...) , ##__VA_ARGS__
#define IMPLEMENTS(...) EXTENDS(__VA_ARGS__)

例如静盅,接口Matcher擴(kuò)展另外一個(gè)抽象接口SelfDescribing,則可以使用EXTENDS寝殴。

struct Description;

INTERFACE(SelfDescribing) {
  virtual void decribeTo(Description&) const = 0;
};

INTERFACE(Matcher) EXTENDS(SelfDescribing) {
  virtual bool matches() const = 0;
};

抽象方法

程序員定義純虛函數(shù)時(shí)蒿叠,特別容易遺忘尾部的= 0的后綴。純虛函數(shù)聲明的是抽象方法蚣常,而虛函數(shù)不能夠準(zhǔn)確表達(dá)這個(gè)語義的市咽。同理,應(yīng)用「代碼生成」的機(jī)制抵蚊,杜絕錯(cuò)誤發(fā)生的可能性施绎。

#define ABSTRACT(...) virtual __VA_ARGS__ = 0

使用ABSTRACT可以增強(qiáng)代碼的可讀性,向用戶聲明這是一個(gè)抽象方法贞绳,而不是純虛函數(shù)谷醉。而且,ABSTRACT置于行首冈闭,而非純虛函數(shù)首尾聲明virtual ... = 0俱尼,相對(duì)更不容易犯錯(cuò)。

INTERFACE(Rule) {
  ABSTRACT(std::string apply(int n) const);
};

覆寫方法

當(dāng)子類覆寫虛函數(shù)時(shí)萎攒,可以使用override增強(qiáng)編譯時(shí)的安全性遇八,及其改善代碼的可讀性。但是耍休,C++11標(biāo)準(zhǔn)的override需要標(biāo)注到函數(shù)簽名的尾部刃永,往往也被程序員遺忘。事實(shí)上羊精,遺忘override并非什么大錯(cuò)斯够,但會(huì)失去編譯時(shí)安全的保護(hù)。

#define OVERRIDE(...) __VA_ARGS__ override

定義OVRIRIDE的準(zhǔn)關(guān)鍵字园匹,將其重要性置于行首雳刺,增強(qiáng)其重要性。

struct Atom : Rule {
  Atom(Matcher* matcher, Action* action);
  
private:
  OVERRIDE(std::string apply(int n) const);
  
private:
  Matcher* matcher;
  Action*  action;  
};

其他關(guān)鍵字

那么是不是應(yīng)該都將C++的其他關(guān)鍵字都定義為宏呢裸违?例如const, constexpr, static掖桦。回答當(dāng)然是"No"供汛,我們只會(huì)定義程序員容易出錯(cuò)的關(guān)鍵場(chǎng)景枪汪,例如涌穆,定義抽象接口時(shí)遺漏虛擬析構(gòu)函數(shù),定義純虛函數(shù)時(shí)遺漏行末的= 0雀久,子類覆寫虛函數(shù)時(shí)遺漏override宿稀。

也許你會(huì)對(duì)使用宏而心有余悸,其實(shí)完全沒必要擔(dān)心赖捌。當(dāng)你習(xí)慣了這套「準(zhǔn)關(guān)鍵字」祝沸,你想犯錯(cuò)都難;而使用原生C++的關(guān)鍵字越庇,我肯定你會(huì)犯錯(cuò)罩锐。

增強(qiáng)的抽象接口

上文定義的cub::details::Interface類,它只定義了虛擬析構(gòu)函數(shù)卤唉。遺憾的是涩惑,在新的C++11標(biāo)準(zhǔn)里,這將阻止Interface自動(dòng)生成移動(dòng)構(gòu)造函數(shù)桑驱,及其移動(dòng)賦值運(yùn)算符竭恬;而且,自動(dòng)生成的拷貝構(gòu)造函數(shù)熬的,及其拷貝賦值運(yùn)算符的規(guī)則也被標(biāo)準(zhǔn)廢棄痊硕,在未來的實(shí)現(xiàn)中可能被拋棄。

那么押框,使用INTERFACE定義的抽象接口寿桨,其實(shí)現(xiàn)該接口的所有子類將不可移動(dòng),只能做保守的拷貝操作强戴。這樣的拷貝行為亭螟,甚至被標(biāo)準(zhǔn)實(shí)現(xiàn)所廢棄钮糖。

因此在C++11的實(shí)現(xiàn)中铺罢,Interface的基類需要進(jìn)行稍許的改進(jìn),使用default顯式聲明所有需要自動(dòng)生成的函數(shù)茴晋。注意道媚,析構(gòu)函數(shù)聲明為public, virtual的扁掸,其他聲明為protected即可。

template <typename T>
struct Interface {
  virtual ~Interface() = default;
  
protected:
  Interface() = default;

  Interface(Interface&&) noexcept = default;
  Interface& operator=(Interface&&) noexcept = default;

  Interface(const Interface&) = default;
  Interface& operator=(const Interface&) = default; 
};
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末最域,一起剝皮案震驚了整個(gè)濱河市谴分,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌镀脂,老刑警劉巖牺蹄,帶你破解...
    沈念sama閱讀 218,546評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異薄翅,居然都是意外死亡沙兰,警方通過查閱死者的電腦和手機(jī)氓奈,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,224評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來鼎天,“玉大人舀奶,你說我怎么就攤上這事≌洌” “怎么了育勺?”我有些...
    開封第一講書人閱讀 164,911評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)罗岖。 經(jīng)常有香客問我怀大,道長(zhǎng),這世上最難降的妖魔是什么呀闻? 我笑而不...
    開封第一講書人閱讀 58,737評(píng)論 1 294
  • 正文 為了忘掉前任,我火速辦了婚禮潜慎,結(jié)果婚禮上捡多,老公的妹妹穿的比我還像新娘。我一直安慰自己铐炫,他們只是感情好垒手,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,753評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著倒信,像睡著了一般科贬。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上鳖悠,一...
    開封第一講書人閱讀 51,598評(píng)論 1 305
  • 那天榜掌,我揣著相機(jī)與錄音,去河邊找鬼乘综。 笑死憎账,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的卡辰。 我是一名探鬼主播胞皱,決...
    沈念sama閱讀 40,338評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼九妈!你這毒婦竟也來了反砌?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,249評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤萌朱,失蹤者是張志新(化名)和其女友劉穎宴树,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體晶疼,經(jīng)...
    沈念sama閱讀 45,696評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡森渐,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,888評(píng)論 3 336
  • 正文 我和宋清朗相戀三年做入,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片同衣。...
    茶點(diǎn)故事閱讀 40,013評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡竟块,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出耐齐,到底是詐尸還是另有隱情浪秘,我是刑警寧澤,帶...
    沈念sama閱讀 35,731評(píng)論 5 346
  • 正文 年R本政府宣布埠况,位于F島的核電站耸携,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏辕翰。R本人自食惡果不足惜夺衍,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,348評(píng)論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望喜命。 院中可真熱鬧沟沙,春花似錦、人聲如沸壁榕。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,929評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽牌里。三九已至颊咬,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間牡辽,已是汗流浹背喳篇。 一陣腳步聲響...
    開封第一講書人閱讀 33,048評(píng)論 1 270
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留态辛,地道東北人杭隙。 一個(gè)月前我還...
    沈念sama閱讀 48,203評(píng)論 3 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像因妙,于是被迫代替她去往敵國(guó)和親痰憎。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,960評(píng)論 2 355