• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            大龍的博客

            常用鏈接

            統(tǒng)計(jì)

            最新評(píng)論

            設(shè)計(jì)模式C++實(shí)現(xiàn)(5)——原型模式、模板方法模式 --- 轉(zhuǎn)

                   軟件領(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è)小例子,加深一下理解。主要參考《大話設(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)。

            1. //父類(lèi)  
            2. class Resume  
            3. {  
            4. protected:  
            5.     char *name;  
            6. public:  
            7.     Resume() {}  
            8.     virtual ~Resume() {}  
            9.     virtual Resume* Clone() { return NULL; }  
            10.     virtual void Set(char *n) {}  
            11.     virtual void Show() {}  
            12. };  
            1. class ResumeA : public Resume  
            2. {  
            3. public:  
            4.     ResumeA(const char *str);  //構(gòu)造函數(shù)  
            5.     ResumeA(const ResumeA &r); //拷貝構(gòu)造函數(shù)  
            6.     ~ResumeA();                //析構(gòu)函數(shù)  
            7.     ResumeA* Clone();          //克隆,關(guān)鍵所在  
            8.     void Show();               //顯示內(nèi)容  
            9. };  
            10. ResumeA::ResumeA(const char *str)   
            11. {  
            12.     if(str == NULL) {  
            13.         name = new char[1];   
            14.         name[0] = '\0';   
            15.     }  
            16.     else {  
            17.         name = new char[strlen(str)+1];  
            18.         strcpy(name, str);  
            19.     }  
            20. }  
            21. ResumeA::~ResumeA() { delete [] name;}  
            22. ResumeA::ResumeA(const ResumeA &r) {  
            23.     name = new char[strlen(r.name)+1];  
            24.     strcpy(name, r.name);  
            25. }  
            26. ResumeA* ResumeA::Clone() {  
            27.     return new ResumeA(*this);  
            28. }  
            29. void ResumeA::Show() {  
            30.     cout<<"ResumeA name : "<<name<<endl;   
            31. }  

                      這里只給出了ResumeA的實(shí)現(xiàn),ResumeB的實(shí)現(xiàn)類(lèi)似。使用的方式如下:

            1. int main()  
            2. {  
            3.     Resume *r1 = new ResumeA("A");  
            4.     Resume *r2 = new ResumeB("B");  
            5.     Resume *r3 = r1->Clone();  
            6.     Resume *r4 = r2->Clone();  
            7.     r1->Show(); r2->Show();  
            8.     //刪除r1,r2  
            9.     delete r1; delete r2;     
            10.     r1 = r2 = NULL;  
            11.     //深拷貝所以對(duì)r3,r4無(wú)影響  
            12.     r3->Show(); r4->Show();  
            13.     delete r3; delete r4;  
            14.     r3 = r4 = NULL;  
            15. }  

                   最近有個(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++代碼。

            1. //簡(jiǎn)歷  
            2. class Resume  
            3. {  
            4. protected: //保護(hù)成員  
            5.     virtual void SetPersonalInfo() {}  
            6.     virtual void SetEducation() {}  
            7.     virtual void SetWorkExp() {}  
            8. public:  
            9.     void FillResume()   
            10.     {  
            11.         SetPersonalInfo();  
            12.         SetEducation();  
            13.         SetWorkExp();  
            14.     }  
            15. };  
            16. class ResumeA: public Resume  
            17. {  
            18. protected:  
            19.     void SetPersonalInfo() { cout<<"A's PersonalInfo"<<endl; }  
            20.     void SetEducation() { cout<<"A's Education"<<endl; }  
            21.     void SetWorkExp() { cout<<"A's Work Experience"<<endl; }  
            22. };  
            23. class ResumeB: public Resume  
            24. {  
            25. protected:  
            26.     void SetPersonalInfo() { cout<<"B's PersonalInfo"<<endl; }  
            27.     void SetEducation() { cout<<"B's Education"<<endl; }  
            28.     void SetWorkExp() { cout<<"B's Work Experience"<<endl; }  
            29. };  

                    使用方式如下:

            1. int main()  
            2. {  
            3.     Resume *r1;  
            4.     r1 = new ResumeA();  
            5.     r1->FillResume();  
            6.     delete r1;  
            7.     r1 = new ResumeB();  
            8.     r1->FillResume();  
            9.     delete r1;  
            10.     r1 = NULL;  
            11.     return 0;  
            12. }  
                       本人享有博客文章的版權(quán),轉(zhuǎn)載請(qǐng)標(biāo)明出處 http://blog.csdn.net/wuzhekai1985

            posted on 2013-09-05 18:46 大龍 閱讀(326) 評(píng)論(0)  編輯 收藏 引用


            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問(wèn)   Chat2DB   管理


            91精品国产91热久久久久福利| 免费无码国产欧美久久18| 久久精品成人欧美大片| 久久久亚洲欧洲日产国码二区| 欧美日韩精品久久久久| 久久久久亚洲精品男人的天堂| 成人久久精品一区二区三区 | 午夜精品久久久内射近拍高清| 99re这里只有精品热久久| 精品国产日韩久久亚洲| 久久久精品国产| 国内精品伊人久久久久网站| 久久本道久久综合伊人| 久久久久99精品成人片| 午夜久久久久久禁播电影| 久久99精品久久久久子伦| 国产精品99久久精品| 久久久久这里只有精品 | 91久久精品电影| 97精品国产97久久久久久免费| 久久精品国产亚洲av麻豆图片| 精品综合久久久久久97超人| 久久精品国产亚洲AV香蕉| 一级女性全黄久久生活片免费| 久久99精品久久久久久hb无码| 91精品日韩人妻无码久久不卡| 漂亮人妻被中出中文字幕久久| 精品久久久久久久中文字幕 | 亚洲中文久久精品无码| 亚洲国产小视频精品久久久三级| 97精品久久天干天天天按摩| 久久97久久97精品免视看秋霞| 91久久香蕉国产熟女线看| 国产成年无码久久久久毛片| 亚洲精品99久久久久中文字幕| 国产激情久久久久影院小草| 欧美丰满熟妇BBB久久久| 久久青草国产精品一区| 久久精品人人做人人爽电影| 亚洲国产精品一区二区久久hs| 97热久久免费频精品99|