1.組合與繼承
1.1組合
1.1.1設(shè)計(jì)模式 Adapter-改造
現(xiàn)在我們?cè)O(shè)計(jì)了一個(gè) 功能很強(qiáng)大的類,但是我們考慮到由于版本發(fā)布的問題(精簡(jiǎn)版,企業(yè)版,旗艦版。。。),我們不可能發(fā)布一個(gè)版本就去開發(fā)一個(gè),這樣成本實(shí)在太高了,仔細(xì)思考以下,既然有了一個(gè)很強(qiáng)大的旗艦版,那么其他削弱的版本從旗艦版刪刪減減不就好了?
這種面向?qū)ο蟮脑O(shè)計(jì)模式就是改造
標(biāo)準(zhǔn)庫(kù)里頭有一個(gè)例子:queue & deque
deque 是兩頭都能出,queue是先進(jìn)先出(只有一個(gè)出口)voidpush(constvalue_type&x){c.push_back(x);}voidpop(){c.pop_front();}};
deque 是兩頭都能出,queue是先進(jìn)先出,deque的功能明顯比queue強(qiáng)大的多,所以queue完全可以由deque實(shí)現(xiàn)。
當(dāng)然改造不只限于刪刪減減,也可以再類里加上其他的類對(duì)象,在現(xiàn)實(shí)生活中我們的汽車都是萬國(guó)牌的,這就好比在汽車這個(gè)類是中有許許多多的其他東西(座椅類,引擎類,安全氣囊類。。。),我們?cè)趯戭惖臅r(shí)候也可以效仿。
在C中,在struct包涵其他的struct就相當(dāng)于改造。
5.1.2復(fù)合下的構(gòu)造和析構(gòu)如何調(diào)用?
構(gòu)造函數(shù)由內(nèi)而外調(diào)用:
容器的構(gòu)造函數(shù)首先調(diào)用組成部分的默認(rèn)構(gòu)造函數(shù),然后執(zhí)行自己的,如果想指定調(diào)用組成的構(gòu)造函數(shù)如下
Container::Container(...):Component(){...};
析構(gòu)函數(shù)由外而內(nèi)調(diào)用:
容器先調(diào)用自己的析構(gòu)函數(shù),在調(diào)用組成的析構(gòu)函數(shù),如果想指定調(diào)用如下
Container::~Container(...){...~Component()};
1.2.1 設(shè)計(jì)模式 Delegation-委托
客戶端+服務(wù)器的架構(gòu)叫C/S架構(gòu),客戶端把信息收集起來發(fā)送到服務(wù)器,由服務(wù)器統(tǒng)一處理,這樣在功能升級(jí)的時(shí)候,只需要在服務(wù)器端的有較大的改動(dòng),而客戶端幾乎不變,例如:有一個(gè)客戶端提供銀行賬號(hào)登錄,后來由于公司業(yè)務(wù)拓展,現(xiàn)在可以支持QQ賬號(hào)登錄,這種改動(dòng)對(duì)應(yīng)客戶端只是用戶改變輸入的賬號(hào)與密碼,主要的改動(dòng)只在服務(wù)器上進(jìn)行。
為了滿足這種設(shè)計(jì)需求,就可以使用 設(shè)計(jì)模式 委托,在類中使用一個(gè)指針指向別的功能類,自身的功能完全借由別的類來完成(可以隨時(shí)改變指向),保持自身面向客戶不變。這種手法也叫編譯防火墻。
現(xiàn)在有a,b,c三個(gè)String類的對(duì)象,同時(shí)共享一個(gè)StringRep對(duì)象:
這個(gè)時(shí)候如果a想要更改內(nèi)容,那么就必須拷貝一份副本給a改,不能影響到b和c。
5.2繼承
Dog屬于動(dòng)物類,從動(dòng)物類繼承:
繼承就是一種傳承,這里涉及到兩個(gè)類,一個(gè)叫父類(基類)另一個(gè)叫子類(派生類),子類可以傳承父類的所有的數(shù)據(jù),還可以對(duì)父類進(jìn)行擴(kuò)展。
繼承的作用在于代碼復(fù)用與擴(kuò)展,繼承后的子類一定是大于等于父類的。
5.2.1繼承下的構(gòu)造與析構(gòu)調(diào)用
繼承中構(gòu)造 和 析構(gòu)的調(diào)用順序:
構(gòu)建子類對(duì)象一定會(huì)先調(diào)用父類的構(gòu)造函數(shù),子類默認(rèn)調(diào)用父類的無參構(gòu)造,當(dāng)然也可以指定調(diào)用父類的構(gòu)造函數(shù),需要通過初始化列表指定調(diào)用
Derived::Derived(...):Base(){...}
析構(gòu)函數(shù)的調(diào)用順序 和 構(gòu)造函數(shù)調(diào)用順序相反
Derived::~Derived(...){...~Base()}
注意:
base class 的析構(gòu)必須是virtual,否則可能會(huì)出現(xiàn)undefined behaviour
1.2.2c++中的繼承方式
(1)公開繼承 class Child:public Parent{};
公開繼承下 父類數(shù)據(jù)到子類之后的權(quán)限變化
父類的公開數(shù)據(jù) 到子類之后是公開的
父類的保護(hù)數(shù)據(jù) 到子類之后是保護(hù)的
父類的私有數(shù)據(jù) 到子類之后是隱藏的
(2)保護(hù)繼承 class Child:protected Parent{};
保護(hù)繼承下 父類數(shù)據(jù) 到子類之后的權(quán)限變換
父類的公開數(shù)據(jù) 到子類之后是保護(hù)的
父類的保護(hù)數(shù)據(jù) 到子類之后是保護(hù)的
父類的私有數(shù)據(jù) 到子類之后是隱藏的
(3)私有繼承 class Child:private Parent{};
私有繼承下 父類數(shù)據(jù) 到子類之后的權(quán)限變換
父類的公開數(shù)據(jù) 到子類之后是私有的
父類的保護(hù)數(shù)據(jù) 到子類之后是私有的
父類的私有數(shù)據(jù) 到子類之后是隱藏的
總結(jié):所謂的繼承方式,就是父類能給子類的最高權(quán)限,實(shí)際權(quán)限小于等于這個(gè)權(quán)限,父類的私有數(shù)據(jù) 到子類之后一定是隱藏的。
class A: B 沒寫權(quán)限就是默認(rèn)私有的
2.虛函數(shù)
Shape是一個(gè)很抽象的概念,下面的長(zhǎng)方形,橢圓是一個(gè)具體的概念,考慮到兩個(gè)圖形都有是一種形狀,為了增加代碼復(fù)用,抽象出Shape這個(gè)類用來描述我們希望圖形對(duì)象所具有的共性。
純虛函數(shù)draw在基類中,這樣就迫使子類里頭必需出現(xiàn)純虛函數(shù)的定義,確保子類圖形能夠被繪制。
虛函數(shù)error,由于不同的圖形可能在操作中會(huì)出現(xiàn)錯(cuò)誤,針對(duì)不同的對(duì)象需要重新定義錯(cuò)誤處理,當(dāng)子類出錯(cuò)那么會(huì)調(diào)用子類對(duì)應(yīng)的error,基類的error提供了一個(gè)通用的錯(cuò)誤處理,如果子類沒有重新定義error,那么就延續(xù)父類的處理方式。
成員函數(shù)objectID,我們希望給創(chuàng)建的所有對(duì)象編號(hào),這不需要子類去重新定義編號(hào)方式,所以使用成員函數(shù)。
2.1運(yùn)行時(shí)綁定
virtual讓類具有多態(tài)性,有的功能函數(shù)不是父類可以實(shí)現(xiàn)的 或者 父類只能提供一個(gè)泛用的方法,那么使用virtual讓方法在子類中實(shí)現(xiàn),這樣在調(diào)用子類時(shí)不會(huì)受到父類的干擾。
在軟件開發(fā)中那些泛用的,誰寫都一樣的東西叫框架,框架大量的使用了virtual,最牛逼的就是MFC(雖然這東西現(xiàn)在名聲不咋滴)。
在C++中支持相關(guān)對(duì)象的不同的成員函數(shù)(原型相同),并允許對(duì)象與適當(dāng)?shù)某蓡T函數(shù)進(jìn)行運(yùn)行時(shí)的綁定,C++通過重寫(overwrite)支持這種機(jī)制----多態(tài)
在上面代碼中我們希望從鍵盤輸入一個(gè)非零的整數(shù)讓程序輸出不同的消息,但是上述代碼無論輸入什么,打印的都是Animal。這是因?yàn)橹羔榩的指向的say函數(shù)是編譯的時(shí)候就確定的。
接著在基類函數(shù)前面加上virtual
現(xiàn)在編譯后,輸出的結(jié)果就和我們從鍵盤輸入的內(nèi)容有關(guān)了。
多態(tài)這種性質(zhì)只發(fā)生在父類型的指針(引用) 指向子類對(duì)象時(shí),通過父類型的指針調(diào)用
虛函數(shù),如果子類重寫了這個(gè)虛函數(shù) ,則調(diào)用的表現(xiàn)是子類的,否則就是父類型中對(duì)應(yīng)的實(shí)現(xiàn)。繼承是構(gòu)成多態(tài)的基礎(chǔ),虛函數(shù)是構(gòu)成多態(tài)的關(guān)鍵 ,函數(shù)重寫是必備條件
2.1.1 重載,覆蓋,隱藏的區(qū)別
(1)重載:
在同一個(gè)類中
函數(shù)名字相同,參數(shù)不同
virtual關(guān)鍵字可有可無
(2)覆蓋:
在不同的類中(基類與派生類)
函數(shù)名字相同,參數(shù)相同
基類必需有virtual關(guān)鍵字
(3)隱藏
在派生類中函數(shù)與基類中的函數(shù)同名,參數(shù)不同,無論有沒有virtual,基類函數(shù)都會(huì)被隱藏
在派生類中函數(shù)與基類中的函數(shù)同名,參數(shù)相同,但是基類沒有virtual,此時(shí)發(fā)生隱藏
2.1.2多態(tài)的應(yīng)用 (類型更加通用 根據(jù)具體的對(duì)象做出具體行為)
用在函數(shù)的參數(shù)上
void testAnimal(Animal* a);
用在函數(shù)的返回值上
Animal* getAnimal(int s);
2.1.3類型識(shí)別
因?yàn)槎鄳B(tài)讓子類對(duì)象富有個(gè)性化,也做到了類型通用,但是通用就意味著失去個(gè)性化,那么在我們?cè)O(shè)計(jì)程序時(shí)可以通過類型識(shí)別恢復(fù)子類的個(gè)性。
2.1.3.1 typeid
typeid這個(gè)運(yùn)算符可以獲得類型或者對(duì)象的類型信息,使用時(shí)需要包涵#include 。
typeid 返回的信息存入一個(gè)type_info 類型的對(duì)象中,這個(gè)類型 重載 == 和 !=運(yùn)算符。并且有個(gè)成員函數(shù) name()返回類型的名稱。
如果父子類之間 沒有多態(tài)性,則當(dāng)父類對(duì)象指針指向子類對(duì)象時(shí),通過指針取值識(shí)別出的對(duì)象是父類型的。
運(yùn)行后輸出結(jié)果如下:
i指的是int,Pi指的是int*,P6Animal指的是Animal*,6Animal指的是Animal。
2.2 虛析構(gòu)
有以下繼承:
我們通過兩種不同的方式創(chuàng)建對(duì)象
(1)
沒有任何問題。
(2)
少了調(diào)用了一次B的析構(gòu)函數(shù),這樣就造成了內(nèi)存泄露。
解決方法:
使用虛析構(gòu)后,資源被釋放。
2.2.1虛表
virtual之所以能呈現(xiàn)出多態(tài)的特性,這是因?yàn)樵诘讓邮褂昧颂摵瘮?shù)表,通過指針連接不同的代碼塊。
如果基類的析構(gòu)函數(shù)沒有被聲明成virtual函數(shù),那么在虛表中就不會(huì)出現(xiàn)析構(gòu)函數(shù),那么class B中的析構(gòu)函數(shù)會(huì)將class A中的析構(gòu)函數(shù)隱藏,然而指針是A*類型,在調(diào)用析構(gòu)的時(shí)候只會(huì)調(diào)用基類的析構(gòu),這樣就導(dǎo)致了內(nèi)存泄露,加上virtual之后,delete時(shí)會(huì)先去調(diào)用子類的析構(gòu)函數(shù),當(dāng)子類的析構(gòu)被調(diào)用,父類的析構(gòu)會(huì)自動(dòng)被調(diào)用,這樣就避免了內(nèi)存泄露。
2.2.2使用指針直接操作虛表
指針是一個(gè)大流氓,得到了地址一切的權(quán)限都只是君子協(xié)議
3.Delegation(委托)+Inheritance(繼承)
設(shè)計(jì)一個(gè)Subject類,其中使用向量容器來保存觀察窗口,同時(shí)需要更新數(shù)據(jù)時(shí)直接調(diào)用set_val函數(shù)。
在寫一個(gè)基類Observer
子類PPT(舉例而已)
主函數(shù)
編譯運(yùn)行后,當(dāng)我們改變了其中一個(gè)的數(shù)據(jù),其余的4個(gè)對(duì)象跟著一起改了,這樣就可以實(shí)現(xiàn)窗口聯(lián)動(dòng)或者一些需要關(guān)聯(lián)的操作,這叫面向?qū)ο笤O(shè)計(jì)。
作者:rrreal
鏈接:http://www.lxweimin.com/p/39abc5336dcf
來源:簡(jiǎn)書
著作權(quán)歸作者所有。商業(yè)轉(zhuǎn)載請(qǐng)聯(lián)系作者獲得授權(quán),非商業(yè)轉(zhuǎn)載請(qǐng)注明出處。