Prototype 模式也正是提供了自我复制的功能, 就是说新对象的创建可以通过已有对象进行创建。在 C++中,拷贝构造函数( Copy Constructor) 曾经是很对程序员的噩梦,浅层拷贝和深层拷贝的魔魇也是很多程序员在面试时候的快餐和系统崩溃时候的根源之一。

在GOF的《设计模式:可复用面向对象软件的基础》中是这样说的:用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。这这个定义中,最重要的一个词是“拷贝”,也就是口头上的复制,而这个拷贝,也就是原型模式的精髓所在。

UML类图

由于克隆需要一个原型,而上面的类图中Prototype就这个原型,Prototype定义了克隆自身的Clone接口,由派生类进行实现,而实现原型模式的重点就在于这个Clone接口的实现。ConcretePrototype1类和ConcretePrototype2类继承自Prototype类,并实现Clone接口,实现克隆自身的操作;同时,在ConcretePrototype1类和ConcretePrototype2类中需要重写默认的复制构造函数,供Clone函数调用,Clone就是通过在内部调用重写的复制构造函数实现的。在后续的编码过程中,如果某个类需要实现Clone功能,就只需要继承Prototype类,然后重写自己的默认复制构造函数就好了。好比在C#中就提供了ICloneable接口,当某个类需要实现原型模式时,只需要实现这个接口的道理是一样的。

使用场合

原型模式和建造者模式、工厂方法模式一样,都属于创建型模式的一种。简单的来说,我们使用原型模式,就是为了创建对象。不过,适合原型模式的最好选择如下:

1.当我们的对象类型不是开始就能确定的,而这个类型是在运行期确定的话,那么我们通过这个类型的对象克隆出一个新的对象比较容易一些;

2.有的时候,我们需要一个对象在某个状态下的副本,此时,我们使用原型模式是最好的选择;例如:一个对象,经过一段处理之后,其内部的状态发生了变化;这个时候,我们需要一个这个状态的副本,如果直接new一个新的对象的话,但是它的状态是不对的,此时,可以使用原型模式,将原来的对象拷贝一个出来,这个对象就和之前的对象是完全一致的了;

3.当我们处理一些比较简单的对象时,并且对象之间的区别很小,可能就几个属性不同而已,那么就可以使用原型模式来完成,省去了创建对象时的麻烦了;

4.有的时候,创建对象时,构造函数的参数很多,而自己又不完全的知道每个参数的意义,就可以使用原型模式来创建一个新的对象,不必去理会创建的过程。

->适当的时候考虑一下原型模式,能减少对应的工作量,减少程序的复杂度,提高效率。

代码实现:

 // clone.cpp : 定义控制台应用程序的入口点。
// #include "stdafx.h"
#include <iostream>
#include<string>
using namespace std; class Prototype
{
public:
Prototype(string str)
{
str_ = str;
}
Prototype()
{
str_ = "";
}
void show()
{
cout<<str_<<endl;
}
virtual Prototype *clone() = ;
private:
string str_;
}; class ConcretePrototype1:public Prototype
{
public:
ConcretePrototype1(string s):Prototype(s){}
ConcretePrototype1(){}
virtual Prototype *clone()
{
ConcretePrototype1 *p = new ConcretePrototype1();
*p = *this;
return p;
}
};
class ConcretePrototype2:public Prototype
{
public:
ConcretePrototype2(string s):Prototype(s){}
ConcretePrototype2(){}
virtual Prototype *clone()
{
ConcretePrototype2 *p = new ConcretePrototype2();
*p = *this;
return p;
}
}; int _tmain(int argc, _TCHAR* argv[])
{
cout<<"原型模式"<<endl;
ConcretePrototype1 *test = new ConcretePrototype1("小李");
test->show();
ConcretePrototype2 *test2 = (ConcretePrototype2 *)test->clone();
test2->show();
system("pause");
return ;
}
 // clonejianli.cpp : 定义控制台应用程序的入口点。
// #include "stdafx.h"
#include <iostream>
#include <string> using namespace std;
class Resume
{
private:
string name;
string sex;
string age;
string timeArea;
string company;
public:
Resume(string s)
{
name = s;
}
void setPersonalInfo(string s,string a)
{
sex = s;
age = a;
}
void setWorkExperience(string t,string c)
{
timeArea = t;
company = c;
}
void Display()
{
cout<<name<<" "<<sex<<" "<<age<<endl;
cout<<"工作经历: "<<timeArea<<" "<<company<<endl<<endl;
}
Resume *clone()
{
Resume *b;
b = new Resume(name);
b->setPersonalInfo(sex,age);
b->setWorkExperience(timeArea,company);
return b;
}
}; int _tmain(int argc, _TCHAR* argv[])
{
Resume *r=new Resume("李俊宏");
r->setPersonalInfo("男","");
r->setWorkExperience("2007-2010","读研究生");
r->Display();
Resume *r2 = r->clone();
r2->setWorkExperience("2003-2007","读本科");
r2->Display();
system("pause");
return ;
}

最新文章

  1. .NET Core 构建配置文件从 project.json 到 .csproj
  2. angular js 的 ng-keyup 监听 keydown keyup事件获取 keyCode
  3. android-Spinner的学习和使用
  4. SQL Server -&gt;&gt; EXECUTE AS LOGIN/USER和Revert表达式
  5. 在本地主机上powershell中连接远程主机执行vbs脚本,得到执行结果(2008版及以上)
  6. [原创]可动态显示圆形图像或圆形文字的AvatarImageView
  7. PHP初学留神(二)
  8. 部署 Graylog 日志系统 - 每天5分钟玩转 Docker 容器技术(92)
  9. adb指令介绍
  10. java日期转化
  11. 从零开始学TensorFlow
  12. docker学习------centos7.5下的swarm集群可视化构建
  13. Java使用DOM4J对XML文件进行增删改查操作
  14. jmeter--001.介绍、下载与安装
  15. 使用 Gradle 构建 Java 项目
  16. java----面对对象
  17. kettle数据库连接使用变量
  18. 关于Segmentation fault (core dumped)
  19. loadrunner&#160;脚本开发-&#160;web_url函数详解
  20. 浅谈Kmeans聚类

热门文章

  1. FWT快速沃尔什变换
  2. 【前端】vue.js实现按钮的动态绑定
  3. Jenkins 安装教程
  4. mysql的一些基本知识
  5. 从零开始玩转JMX(二)——Condition
  6. Windows 下 ORA-12560: TNS: 协议适配器错误的问题
  7. Win10m的前景到底在何方?
  8. [Network Architecture]DPN(Dual Path Network)算法详解(转)
  9. Flume实例一学习
  10. LA 5846 霓虹灯广告牌(单色三角形问题)