(C++实现)——原型模式(Prototype Pattern)

2024-06-15 02:58

本文主要是介绍(C++实现)——原型模式(Prototype Pattern),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

解决的问题:

用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。这个其实和C++的拷贝构造函数的作用是一致的,实际上就是动态抽取当前对象运行时的状态。


类图结构:     

客户(Client)角色:客户类提出创建对象的请求。
抽象原型(Prototype)角色:这是一个抽象角色,通常由一个C#接口或抽象类实现。此角色给出所有的具体原型类所需的接口。在C#中,抽象原型角色通常实现了ICloneable接口。
具体原型(Concrete Prototype)角色:被复制的对象。此角色需要实现抽象原型角色所要求的接口。


样例实现:

例子参照wuzhekai1985的简历的例子,代码拷贝如下:

[cpp]  view plain copy
  1. // CplusplusPrototype.cpp : Defines the entry point for the console application.  
  2. //  
  3. #include "stdafx.h"  
  4. #include<iostream>  
  5. #include<vector>  
  6. #include<assert.h>  
  7. using namespace std;  
  8. //父类    
  9. class Resume    
  10. {    
  11. protected:    
  12.     char *name;    
  13. public:    
  14.     Resume() {}    
  15.     virtual ~Resume() {}    
  16.     virtual Resume* Clone() { return NULL; }    
  17.     virtual void Set(char *n) {}    
  18.     virtual void Show() {}    
  19. };   
  20. class ResumeA : public Resume    
  21. {    
  22. public:    
  23.     ResumeA(const char *str);  //构造函数    
  24.     ResumeA(const ResumeA &r); //拷贝构造函数    
  25.     ~ResumeA();                //析构函数    
  26.     ResumeA* Clone();          //克隆,关键所在    
  27.     void Show();               //显示内容    
  28. };    
  29. ResumeA::ResumeA(const char *str)     
  30. {    
  31.     if(str == NULL) {    
  32.         name = new char[1];     
  33.         name[0] = '\0';     
  34.     }    
  35.     else {    
  36.         name = new char[strlen(str)+1];    
  37.         strcpy(name, str);    
  38.     }    
  39. }    
  40. ResumeA::~ResumeA() { delete [] name;}    
  41. ResumeA::ResumeA(const ResumeA &r) {    
  42.     name = new char[strlen(r.name)+1];    
  43.     strcpy(name, r.name);    
  44. }    
  45. ResumeA* ResumeA::Clone() {    
  46.     return new ResumeA(*this);    
  47. }    
  48. void ResumeA::Show() {    
  49.     cout<<"ResumeA name : "<<name<<endl;     
  50. }   
  51.   
  52. class ResumeB : public Resume    
  53. {    
  54. public:    
  55.     ResumeB(const char *str);  //构造函数    
  56.     ResumeB(const ResumeB &r); //拷贝构造函数    
  57.     ~ResumeB();                //析构函数    
  58.     ResumeB* Clone();          //克隆,关键所在    
  59.     void Show();               //显示内容    
  60. };    
  61. ResumeB::ResumeB(const char *str)     
  62. {    
  63.     if(str == NULL) {    
  64.         name = new char[1];     
  65.         name[0] = '\0';     
  66.     }    
  67.     else {    
  68.         name = new char[strlen(str)+1];    
  69.         strcpy(name, str);    
  70.     }    
  71. }    
  72. ResumeB::~ResumeB() { delete [] name;}    
  73. ResumeB::ResumeB(const ResumeB &r) {    
  74.     name = new char[strlen(r.name)+1];    
  75.     strcpy(name, r.name);    
  76. }    
  77. ResumeB* ResumeB::Clone() {    
  78.     return new ResumeB(*this);    
  79. }    
  80. void ResumeB::Show() {    
  81.     cout<<"ResumeB name : "<<name<<endl;     
  82. }   
  83.   
  84. int _tmain(int argc, _TCHAR* argv[])  
  85. {  
  86.     Resume *r1 = new ResumeA("A");    
  87.     Resume *r2 = new ResumeB("B");    
  88.     Resume *r3 = r1->Clone();    
  89.     Resume *r4 = r2->Clone();    
  90.     r1->Show(); r2->Show();    
  91.     //删除r1,r2    
  92.     delete r1; delete r2;       
  93.     r1 = r2 = NULL;    
  94.     //深拷贝所以对r3,r4无影响    
  95.     r3->Show(); r4->Show();    
  96.     delete r3; delete r4;    
  97.     r3 = r4 = NULL;       
  98.     return 0;  
  99. }  


带Prototype Manager的原型模式:

客户(Client)角色:客户端类向原型管理器提出创建对象的请求。
抽象原型(Prototype)角色:这是一个抽象角色,通常由一个C#接口或抽象类实现。此角色给出所有的具体原型类所需的接口。在C#中,抽象原型角色通常实现了ICloneable接口。
具体原型(Concrete Prototype)角色:被复制的对象。此角色需要实现抽象的原型角色所要求的接口。
原型管理器(Prototype Manager)角色:创建具体原型类的对象,并记录每一个被创建的对象。

代码实现如下:

[cpp]  view plain copy
  1. // CplusplusPrototype.cpp : Defines the entry point for the console application.  
  2. //  
  3.   
  4. #include "stdafx.h"  
  5. #include<iostream>  
  6. #include<vector>  
  7. #include<assert.h>  
  8. using namespace std;  
  9. //父类    
  10. class Resume    
  11. {    
  12. protected:    
  13.     char *name;    
  14. public:    
  15.     Resume() {}    
  16.     virtual ~Resume() {}    
  17.     virtual Resume* Clone() { return NULL; }    
  18.     virtual void Set(char *n) {}    
  19.     virtual void Show() {}    
  20. };   
  21. class ResumeA : public Resume    
  22. {    
  23. public:    
  24.     ResumeA(const char *str);  //构造函数    
  25.     ResumeA(const ResumeA &r); //拷贝构造函数    
  26.     ~ResumeA();                //析构函数    
  27.     ResumeA* Clone();          //克隆,关键所在    
  28.     void Show();               //显示内容    
  29. };    
  30. ResumeA::ResumeA(const char *str)     
  31. {    
  32.     if(str == NULL) {    
  33.         name = new char[1];     
  34.         name[0] = '\0';     
  35.     }    
  36.     else {    
  37.         name = new char[strlen(str)+1];    
  38.         strcpy(name, str);    
  39.     }    
  40. }    
  41. ResumeA::~ResumeA() { delete [] name;}    
  42. ResumeA::ResumeA(const ResumeA &r) {    
  43.     name = new char[strlen(r.name)+1];    
  44.     strcpy(name, r.name);    
  45. }    
  46. ResumeA* ResumeA::Clone() {    
  47.     return new ResumeA(*this);    
  48. }    
  49. void ResumeA::Show() {    
  50.     cout<<"ResumeA name : "<<name<<endl;     
  51. }   
  52.   
  53. class ResumeB : public Resume    
  54. {    
  55. public:    
  56.     ResumeB(const char *str);  //构造函数    
  57.     ResumeB(const ResumeB &r); //拷贝构造函数    
  58.     ~ResumeB();                //析构函数    
  59.     ResumeB* Clone();          //克隆,关键所在    
  60.     void Show();               //显示内容    
  61. };    
  62. ResumeB::ResumeB(const char *str)     
  63. {    
  64.     if(str == NULL) {    
  65.         name = new char[1];     
  66.         name[0] = '\0';     
  67.     }    
  68.     else {    
  69.         name = new char[strlen(str)+1];    
  70.         strcpy(name, str);    
  71.     }    
  72. }    
  73. ResumeB::~ResumeB() { delete [] name;}    
  74. ResumeB::ResumeB(const ResumeB &r) {    
  75.     name = new char[strlen(r.name)+1];    
  76.     strcpy(name, r.name);    
  77. }    
  78. ResumeB* ResumeB::Clone() {    
  79.     return new ResumeB(*this);    
  80. }    
  81. void ResumeB::Show() {    
  82.     cout<<"ResumeB name : "<<name<<endl;     
  83. }   
  84.   
  85. class ResumeManager  
  86. {  
  87. private:  
  88.     vector<Resume *> mResume;  
  89. public:  
  90.     ResumeManager()  
  91.     {  
  92.   
  93.     }  
  94.     void add(Resume * resume)  
  95.     {  
  96.         mResume.push_back(resume);  
  97.     }  
  98.   
  99.      Resume * get(int index) const  
  100.     {  
  101.         assert(index>=0 && index<mResume.size());  
  102.         return mResume[index];  
  103.     }  
  104. };  
  105.   
  106. int _tmain(int argc, _TCHAR* argv[])  
  107. {  
  108.     ResumeManager *manager = new ResumeManager();  
  109.     Resume *r1 = new ResumeA("A");    
  110.     Resume *r2 = new ResumeB("B");    
  111.     manager->add(r1);  
  112.     manager->add(r2);  
  113.     manager->get(0)->Show();   
  114.     manager->get(1)->Show();    
  115.     Resume *r3 = manager->get(0)->Clone();    
  116.     Resume *r4 = manager->get(1)->Clone();   
  117.   
  118.     //删除r1,r2    
  119.     delete r1; delete r2;       
  120.     r1 = r2 = NULL;    
  121.     //深拷贝所以对r3,r4无影响    
  122.     r3->Show(); r4->Show();    
  123.     delete r3; delete r4;    
  124.     r3 = r4 = NULL;   
  125.     return 0;  
  126. }  

实现要点:

1.使用原型管理器,体现在一个系统中原型数目不固定时,可以动态的创建和销毁。

2.实现克隆操作,在.NET中可以使用Object类的MemberwiseClone()方法来实现对象的浅表拷贝或通过序列化的方式来实现深拷贝,在C++中就是拷贝构造函数的作用。

3.Prototype模式同样用于隔离类对象的使用者和具体类型(易变类)之间的耦合关系,它同样要求这些“易变类”拥有稳定的接口。


效果:

1.它对客户隐藏了具体的产品类,因此减少了客户知道的名字的数目。

2. Prototype模式允许客户只通过注册原型实例就可以将一个具体产品类并入到系统中,客户可以在运行时刻建立和删除原型。

3.减少了子类构造,Prototype模式是克隆一个原型而不是请求工厂方法创建一个,所以它不需要一个与具体产品类平行的Creater类层次。

4.Portotype模式具有给一个应用软件动态加载新功能的能力。由于Prototype的独立性较高,可以很容易动态加载新功能而不影响老系统。

5.产品类不需要非得有任何事先确定的等级结构,因为Prototype模式适用于任何的等级结构

6.Prototype模式的最主要缺点就是每一个类必须配备一个克隆方法。而且这个克隆方法需要对类的功能进行通盘考虑,这对全新的类来说不是很难,但对已有的类进行改造时,不一定是件容易的事。


适用性:

1. 当一个系统应该独立于他的产品创建、构成和表示时,需要使用原型模式

2. 当要实例化的类是在运行时刻指定时,如通过动态装载

3. 为了避免创建一个与产品类层次平行的工厂类层次时

4. 当一个类的实例只能有几个不同状态组合中的一种时,建立相应数目的原型并克隆他们可能比每次用合适的状态手工实例化该类更方便一些。



LCL_data原创于CSDN.NET【http://blog.csdn.net/lcl_data/article/details/8764228】

这篇关于(C++实现)——原型模式(Prototype Pattern)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1062244

相关文章

使用Python实现在Word中添加或删除超链接

《使用Python实现在Word中添加或删除超链接》在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能,本文将为大家介绍一下Python如何实现在Word中添加或... 在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能。通过添加超

windos server2022里的DFS配置的实现

《windosserver2022里的DFS配置的实现》DFS是WindowsServer操作系统提供的一种功能,用于在多台服务器上集中管理共享文件夹和文件的分布式存储解决方案,本文就来介绍一下wi... 目录什么是DFS?优势:应用场景:DFS配置步骤什么是DFS?DFS指的是分布式文件系统(Distr

NFS实现多服务器文件的共享的方法步骤

《NFS实现多服务器文件的共享的方法步骤》NFS允许网络中的计算机之间共享资源,客户端可以透明地读写远端NFS服务器上的文件,本文就来介绍一下NFS实现多服务器文件的共享的方法步骤,感兴趣的可以了解一... 目录一、简介二、部署1、准备1、服务端和客户端:安装nfs-utils2、服务端:创建共享目录3、服

C#使用yield关键字实现提升迭代性能与效率

《C#使用yield关键字实现提升迭代性能与效率》yield关键字在C#中简化了数据迭代的方式,实现了按需生成数据,自动维护迭代状态,本文主要来聊聊如何使用yield关键字实现提升迭代性能与效率,感兴... 目录前言传统迭代和yield迭代方式对比yield延迟加载按需获取数据yield break显式示迭

Python实现高效地读写大型文件

《Python实现高效地读写大型文件》Python如何读写的是大型文件,有没有什么方法来提高效率呢,这篇文章就来和大家聊聊如何在Python中高效地读写大型文件,需要的可以了解下... 目录一、逐行读取大型文件二、分块读取大型文件三、使用 mmap 模块进行内存映射文件操作(适用于大文件)四、使用 pand

python实现pdf转word和excel的示例代码

《python实现pdf转word和excel的示例代码》本文主要介绍了python实现pdf转word和excel的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价... 目录一、引言二、python编程1,PDF转Word2,PDF转Excel三、前端页面效果展示总结一

Python xmltodict实现简化XML数据处理

《Pythonxmltodict实现简化XML数据处理》Python社区为提供了xmltodict库,它专为简化XML与Python数据结构的转换而设计,本文主要来为大家介绍一下如何使用xmltod... 目录一、引言二、XMLtodict介绍设计理念适用场景三、功能参数与属性1、parse函数2、unpa

C#实现获得某个枚举的所有名称

《C#实现获得某个枚举的所有名称》这篇文章主要为大家详细介绍了C#如何实现获得某个枚举的所有名称,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考一下... C#中获得某个枚举的所有名称using System;using System.Collections.Generic;usi

Go语言实现将中文转化为拼音功能

《Go语言实现将中文转化为拼音功能》这篇文章主要为大家详细介绍了Go语言中如何实现将中文转化为拼音功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 有这么一个需求:新用户入职 创建一系列账号比较麻烦,打算通过接口传入姓名进行初始化。想把姓名转化成拼音。因为有些账号即需要中文也需要英

C# 读写ini文件操作实现

《C#读写ini文件操作实现》本文主要介绍了C#读写ini文件操作实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录一、INI文件结构二、读取INI文件中的数据在C#应用程序中,常将INI文件作为配置文件,用于存储应用程序的