• <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>

            大龍的博客

            常用鏈接

            統計

            最新評論

            設計模式C++實現(5)——原型模式、模板方法模式 --- 轉

                   軟件領域中的設計模式為開發人員提供了一種使用專家設計經驗的有效途徑。設計模式中運用了面向對象編程語言的重要特性:封裝、繼承、多態,真正領悟設計 模式的精髓是可能一個漫長的過程,需要大量實踐經驗的積累。最近看設計模式的書,對于每個模式,用C++寫了個小例子,加深一下理解。主要參考《大話設計 模式》和《設計模式:可復用面向對象軟件的基礎》(DP)兩本書。本文介紹原型模式和模板方法模式的實現。首先介紹原型模式,然后引出模板方法模式。

                   DP書上的定義為:用原型實例指定創建對象的種類,并且通過拷貝這些原型創建新的對象。其中有一個詞很重要,那就是拷貝??梢哉f,拷貝是原型模式的精髓 所在。舉個現實中的例子來介紹原型模式。找工作的時候,我們需要準備簡歷。假設沒有打印設備,因此需手寫簡歷,這些簡歷的內容都是一樣的。這樣有個缺陷, 如果要修改簡歷中的某項,那么所有已寫好的簡歷都要修改,工作量很大。隨著科技的進步,出現了打印設備。我們只需手寫一份,然后利用打印設備復印多份即 可。如果要修改簡歷中的某項,那么修改原始的版本就可以了,然后再復印。原始的那份手寫稿相當于是一個原型,有了它,就可以通過復?。截悾﹦撛斐龈嗟?新簡歷。這就是原型模式的基本思想。下面給出原型模式的UML圖,以剛才那個例子為實例。


                    原型模式實現的關鍵就是實現Clone函數,對于C++來說,其實就是拷貝構造函數,需實現深拷貝,下面給出一種實現。

            1. //父類  
            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);  //構造函數  
            5.     ResumeA(const ResumeA &r); //拷貝構造函數  
            6.     ~ResumeA();                //析構函數  
            7.     ResumeA* Clone();          //克隆,關鍵所在  
            8.     void Show();               //顯示內容  
            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的實現,ResumeB的實現類似。使用的方式如下:

            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.     //深拷貝所以對r3,r4無影響  
            12.     r3->Show(); r4->Show();  
            13.     delete r3; delete r4;  
            14.     r3 = r4 = NULL;  
            15. }  

                   最近有個招聘會,可以帶上簡歷去應聘了。但是,其中有一家公司不接受簡歷,而是給應聘者發了一張簡歷表,上面有基本信息、教育背景、工作經歷等欄,讓應 聘者按照要求填寫完整。每個人拿到這份表格后,就開始填寫。如果用程序實現這個過程,該如何做呢?一種方案就是用模板方法模式:定義一個操作中的算法的骨架,而將一些步驟延遲到子類中。模板方法使得子類可以不改變一個算法的結構即可重定義該算法的某些特定步驟。我們的例子中,操作就是填寫簡歷這一過程,我們可以在父類中定義操作的算法骨架,而具體的實現由子類完成。下面給出它的UML圖。

                   其中FillResume() 定義了操作的骨架,依次調用子類實現的函數。相當于每個人填寫簡歷的實際過程。接著給出相應的C++代碼。

            1. //簡歷  
            2. class Resume  
            3. {  
            4. protected: //保護成員  
            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. }  
                       本人享有博客文章的版權,轉載請標明出處 http://blog.csdn.net/wuzhekai1985

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

            久久亚洲AV成人出白浆无码国产| 韩国免费A级毛片久久| 丰满少妇人妻久久久久久4| 91精品国产91久久久久久青草| 2021精品国产综合久久| 精品久久久久久久久午夜福利| 91精品国产乱码久久久久久| 国产伊人久久| 久久综合狠狠综合久久综合88| 99久久99这里只有免费费精品| 91精品国产91久久| 香蕉久久久久久狠狠色| 久久精品国产亚洲AV高清热 | 中文字幕人妻色偷偷久久| 久久人人爽人人爽人人片AV不 | 久久中文字幕人妻熟av女| 色欲av伊人久久大香线蕉影院| 99久久精品免费国产大片| 久久人人爽人人人人片av| 国产成人精品久久一区二区三区av| 国色天香久久久久久久小说| 久久久久国色AV免费观看| 99久久免费国产精品热| 日韩人妻无码一区二区三区久久| 麻豆久久| 久久影院亚洲一区| 久久精品国产亚洲一区二区| 人妻精品久久无码专区精东影业| 午夜精品久久久久久| 国产ww久久久久久久久久| 久久精品国产久精国产思思 | 日韩精品国产自在久久现线拍| 亚洲国产精品18久久久久久| 国产69精品久久久久观看软件| 亚洲乱码日产精品a级毛片久久 | 一本色道久久88—综合亚洲精品| 久久久久久无码国产精品中文字幕| 久久国产精品99久久久久久老狼| 人妻精品久久久久中文字幕69| 中文字幕人妻色偷偷久久| 久久精品中文字幕一区|