生活随笔
收集整理的這篇文章主要介紹了
设计模式C++实现(5)——原型模式、模板方法模式
小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.
?軟件領(lǐng)域中的設(shè)計(jì)模式為開(kāi)發(fā)人員提供了一種使用專(zhuān)家設(shè)計(jì)經(jīng)驗(yàn)的有效途徑。設(shè)計(jì)模式中運(yùn)用了面向?qū)ο缶幊陶Z(yǔ)言的重要特性:封裝、繼承、多態(tài),真正領(lǐng)悟設(shè)計(jì)模式的精髓是可能一個(gè)漫長(zhǎng)的過(guò)程,需要大量實(shí)踐經(jīng)驗(yàn)的積累。最近看設(shè)計(jì)模式的書(shū),對(duì)于每個(gè)模式,用C++寫(xiě)了個(gè)小例子,加深一下理解。主要參考《大話(huà)設(shè)計(jì)模式》和《設(shè)計(jì)模式:可復(fù)用面向?qū)ο筌浖幕A(chǔ)》(DP)兩本書(shū)。本文介紹原型模式和模板方法模式的實(shí)現(xiàn)。首先介紹原型模式,然后引出模板方法模式。
? ? ? ?DP書(shū)上的定義為:用原型實(shí)例指定創(chuàng)建對(duì)象的種類(lèi),并且通過(guò)拷貝這些原型創(chuàng)建新的對(duì)象。其中有一個(gè)詞很重要,那就是拷貝。可以說(shuō),拷貝是原型模式的精髓所在。舉個(gè)現(xiàn)實(shí)中的例子來(lái)介紹原型模式。找工作的時(shí)候,我們需要準(zhǔn)備簡(jiǎn)歷。假設(shè)沒(méi)有打印設(shè)備,因此需手寫(xiě)簡(jiǎn)歷,這些簡(jiǎn)歷的內(nèi)容都是一樣的。這樣有個(gè)缺陷,如果要修改簡(jiǎn)歷中的某項(xiàng),那么所有已寫(xiě)好的簡(jiǎn)歷都要修改,工作量很大。隨著科技的進(jìn)步,出現(xiàn)了打印設(shè)備。我們只需手寫(xiě)一份,然后利用打印設(shè)備復(fù)印多份即可。如果要修改簡(jiǎn)歷中的某項(xiàng),那么修改原始的版本就可以了,然后再?gòu)?fù)印。原始的那份手寫(xiě)稿相當(dāng)于是一個(gè)原型,有了它,就可以通過(guò)復(fù)印(拷貝)創(chuàng)造出更多的新簡(jiǎn)歷。這就是原型模式的基本思想。下面給出原型模式的UML圖,以剛才那個(gè)例子為實(shí)例。
? ? ? ? 原型模式實(shí)現(xiàn)的關(guān)鍵就是實(shí)現(xiàn)Clone函數(shù),對(duì)于C++來(lái)說(shuō),其實(shí)就是拷貝構(gòu)造函數(shù),需實(shí)現(xiàn)深拷貝,下面給出一種實(shí)現(xiàn)。
[cpp]?view plaincopy print?
?? class?Resume?? {?? protected:?? ????char?*name;?? public:?? ????Resume()?{}?? ????virtual?~Resume()?{}?? ????virtual?Resume*?Clone()?{?return?NULL;?}?? ????virtual?void?Set(char?*n)?{}?? ????virtual?void?Show()?{}?? };??
[cpp]?view plaincopy print?
class?ResumeA?:?public?Resume?? {?? public:?? ????ResumeA(const?char?*str);???? ????ResumeA(const?ResumeA?&r);??? ????~ResumeA();?????????????????? ????ResumeA*?Clone();???????????? ????void?Show();????????????????? };?? ResumeA::ResumeA(const?char?*str)??? {?? ????if(str?==?NULL)?{?? ????????name?=?new?char[1];??? ????????name[0]?=?'\0';??? ????}?? ????else?{?? ????????name?=?new?char[strlen(str)+1];?? ????????strcpy(name,?str);?? ????}?? }?? ResumeA::~ResumeA()?{?delete?[]?name;}?? ResumeA::ResumeA(const?ResumeA?&r)?{?? ????name?=?new?char[strlen(r.name)+1];?? ????strcpy(name,?r.name);?? }?? ResumeA*?ResumeA::Clone()?{?? ????return?new?ResumeA(*this);?? }?? void?ResumeA::Show()?{?? ????cout<<"ResumeA?name?:?"<<name<<endl;??? }??
? ? ? ? ? 這里只給出了ResumeA的實(shí)現(xiàn),ResumeB的實(shí)現(xiàn)類(lèi)似。使用的方式如下:
[cpp]?view plaincopy print?
int?main()?? {?? ????Resume?*r1?=?new?ResumeA("A");?? ????Resume?*r2?=?new?ResumeB("B");?? ????Resume?*r3?=?r1->Clone();?? ????Resume?*r4?=?r2->Clone();?? ????r1->Show();?r2->Show();?? ?????? ????delete?r1;?delete?r2;????? ????r1?=?r2?=?NULL;?? ?????? ????r3->Show();?r4->Show();?? ????delete?r3;?delete?r4;?? ????r3?=?r4?=?NULL;?? }??
? ? ? ?最近有個(gè)招聘會(huì),可以帶上簡(jiǎn)歷去應(yīng)聘了。但是,其中有一家公司不接受簡(jiǎn)歷,而是給應(yīng)聘者發(fā)了一張簡(jiǎn)歷表,上面有基本信息、教育背景、工作經(jīng)歷等欄,讓?xiě)?yīng)聘者按照要求填寫(xiě)完整。每個(gè)人拿到這份表格后,就開(kāi)始填寫(xiě)。如果用程序?qū)崿F(xiàn)這個(gè)過(guò)程,該如何做呢?一種方案就是用模板方法模式:定義一個(gè)操作中的算法的骨架,而將一些步驟延遲到子類(lèi)中。模板方法使得子類(lèi)可以不改變一個(gè)算法的結(jié)構(gòu)即可重定義該算法的某些特定步驟。我們的例子中,操作就是填寫(xiě)簡(jiǎn)歷這一過(guò)程,我們可以在父類(lèi)中定義操作的算法骨架,而具體的實(shí)現(xiàn)由子類(lèi)完成。下面給出它的UML圖。
? ? ? ?其中FillResume() 定義了操作的骨架,依次調(diào)用子類(lèi)實(shí)現(xiàn)的函數(shù)。相當(dāng)于每個(gè)人填寫(xiě)簡(jiǎn)歷的實(shí)際過(guò)程。接著給出相應(yīng)的C++代碼。
[cpp]?view plaincopy print?
?? class?Resume?? {?? protected:??? ????virtual?void?SetPersonalInfo()?{}?? ????virtual?void?SetEducation()?{}?? ????virtual?void?SetWorkExp()?{}?? public:?? ????void?FillResume()??? ????{?? ????????SetPersonalInfo();?? ????????SetEducation();?? ????????SetWorkExp();?? ????}?? };?? class?ResumeA:?public?Resume?? {?? protected:?? ????void?SetPersonalInfo()?{?cout<<"A's?PersonalInfo"<<endl;?}?? ????void?SetEducation()?{?cout<<"A's?Education"<<endl;?}?? ????void?SetWorkExp()?{?cout<<"A's?Work?Experience"<<endl;?}?? };?? class?ResumeB:?public?Resume?? {?? protected:?? ????void?SetPersonalInfo()?{?cout<<"B's?PersonalInfo"<<endl;?}?? ????void?SetEducation()?{?cout<<"B's?Education"<<endl;?}?? ????void?SetWorkExp()?{?cout<<"B's?Work?Experience"<<endl;?}?? };??
? ? ? ? 使用方式如下:
[cpp]?view plaincopy print?
int?main()?? {?? ????Resume?*r1;?? ????r1?=?new?ResumeA();?? ????r1->FillResume();?? ????delete?r1;?? ????r1?=?new?ResumeB();?? ????r1->FillResume();?? ????delete?r1;?? ????r1?=?NULL;?? ????return?0;?? }??
? ? ? ? ?
總結(jié)
以上是生活随笔為你收集整理的设计模式C++实现(5)——原型模式、模板方法模式的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。