设计模式(4)-对象创建型模式-Prototype模式

1.对象创建型模式

1.4          Protoype模式

1.4.1需求

通过拷贝原形对象创建新的对象。

1.4.2结构

?P r o t o t y p e(Gr a p h i c)

— 声明一个克隆自身的接口。

?C o n c r e t e P r o t o t y p e(S t a ff、W h o l e N o t e、H a l fN o t e)

— 实现一个克隆自身的操作。

?  C l i e n t(G r a p h i c To o l)

— 让一个原型克隆自身从而创建一个新的对象。

1.4.3样例-C++

//Prototype.h

#ifndef _PROTOTYPE_H_

#define _PROTOTYPE_H_

class Prototype

{

public:

virtual ~Prototype();

virtual Prototype*Clone()const = 0;

protected:

Prototype();

private:

};

class ConcretePrototype:public Prototype

{

public:

ConcretePrototype();

ConcretePrototype(const  ConcretePrototype& cp);

~ConcretePrototype();

Prototype* Clone() const;

protected:

private:

};

#endif //~_PROTOTYPE_H_

代码片断2:Prototype.cpp

//Prototype.cpp

#include"Prototype.h"

#include<iostream>

using namespace std;

Prototype::Prototype()

{

}

Prototype::~Prototype()

{

}

Prototype* Prototype::Clone() const

{

return 0;

}

ConcretePrototype::ConcretePrototype()

{

}

ConcretePrototype::~ConcretePrototype()

{

}

ConcretePrototype::ConcretePrototype(const  ConcretePrototype& cp)

{

cout<<"ConcretePrototype copy..."<<endl;

}

Prototype* ConcretePrototype::Clone() const

{

return newConcretePrototype(*this);

}

//main.cpp

#include"Prototype.h"

#include<iostream>

using namespace std;

int main(int argc,char*argv[])

{

Prototype* p= newConcretePrototype();

Prototype* p1=p->Clone();

return 0;

}

注:这里仅仅是说明概念,没有涉及C++常见的深度拷贝问题.

1.4.4 样例-JAVA

在Java中,原型模式能够非常easy地实现,仅仅要实现Cloneable这个标识性的接口,再覆盖该接口中的clone()方法,就可以“克隆”该实现类的不论什么一个对象。

class ConcretePrototype02 implements Cloneable{

private String name;

private ArrayList<String> nameList = new ArrayList<String>();

public ConcretePrototype02(String name) {

this.name = name;

this.nameList.add(this.name);

}

//加入nameList中的对象

public void setName(String name) {

this.nameList.add(name);

}

public ArrayList<String> getNameList() {

return this.nameList;

}

//覆盖Object基类中的clone()方法,并扩大该方法的訪问权限,详细化返回本类型

public ConcretePrototype02 clone() {

ConcretePrototype02self = null;

try {

self= (ConcretePrototype02) super.clone();

//下面这句是实现深拷贝的关键

//      self.nameList =(ArrayList<String>) this.nameList.clone();

} catch (CloneNotSupportedException e) {

e.printStackTrace();

}

return self;

}

}

//測试类

public class Client {

public static void main(String[] args) {

ConcretePrototype02prototype02 = new ConcretePrototype02("蚂蚁 ...");

System.out.println(prototype02.getNameList());

//通过clone获得一个拷贝

ConcretePrototype02fromClone02 = prototype02.clone();

fromClone02.setName("小蚂蚁 ...");

System.out.println(fromClone02.getNameList());

System.out.println(prototype02.getNameList());

}

}

測试结果:


拷贝之前的原型: [蚂蚁 ...]

拷贝得到的对象: [蚂蚁 ..., 小蚂蚁 ...]

拷贝之后的原型: [蚂蚁 ..., 小蚂蚁 ...]

发现拷贝之后原来的对象持有的ArrayList<String>类型的nameList引用会随着拷贝得到的fromClone对象运行了setName()方法而改变。这不是我们想要的结果,由于这意味着原型以及拷贝得到的对象共享同一个引用变量,这是线程不安全的。当我们去掉   
上面clone()方法中被凝视的语句之后再測试,得到结果例如以下:


拷贝之前的原型: [蚂蚁 ...]

拷贝得到的对象: [蚂蚁 ..., 小蚂蚁 ...]

拷贝之后的原型: [蚂蚁 ...]

在Java中使用原型模式Prototype是相当简单的,仅仅要记住几点注意点,就能够方便地实现该模式了。

因为使用clone()方法来拷贝一个对象是从内存二进制流中进行IO读写。所以拷贝得到一个对象是不会运行该对象所相应类的构造函数的。

总结例如以下:

1、构造函数不会被运行。

2、类的成员变量中若有引用类型的变量(数组也是一种对象)。默认的clone()并不会对其进行拷贝,需自行提供深拷贝;

时间: 2024-08-08 09:41:32

设计模式(4)-对象创建型模式-Prototype模式的相关文章

设计模式(3)-对象创建型模式-Abstract Factory模式

1.对象创建型模式 1.3           Abstract Factory模式 1.3.1 需求 在下面情况能够使用Abstract Factory模式: ?  一个系统要独立于它的产品的创建.组合和表示时(这个需求和FactoryMethod类似). ?  一个系统要由多个产品系列中的一个来配置时(这个需求也和Factory Method类似). ?  当你要强调一系列相关的产品对象的设计以便进行联合使用时(这个需求表明一个工厂要创建多个相关的产品对象,是比FactoryMethod多的

设计模式(创建型)之原型模式(Prototype Pattern)

PS一句:最终还是选择CSDN来整理发表这几年的知识点,该文章平行迁移到CSDN.因为CSDN也支持MarkDown语法了,牛逼啊! 概述 原型模式是一种创建型设计模式,它通过复制一个已经存在的实例来返回新的实例,而不是新建实例.被复制的实例就是我们所称的原型,这个原型是可定制的.原型模式多用于创建复杂的或者耗时的实例, 因为这种情况下,复制一个已经存在的实例可以使程序运行更高效,或者创建值相等,只是命名不一样的同类数据. 原型模式要求对象实现一个可以"克隆"自身的接口,这样就可以通过

天天设计模式二:创建型模式实践

天天设计模式二:创建型模式实践 创建型设计模式主要应用在对象创建,以不同的方式来满足系统的灵活性配置.动态注入.等应用场景. 一.单例模式 二.抽象工厂 三.建造者模式 四.工厂方法模式 五.原型模式

设计模式(创建型)之建造者模式(Builder Pattern)

PS一句:最终还是选择CSDN来整理发表这几年的知识点,该文章平行迁移到CSDN.因为CSDN也支持MarkDown语法了,牛逼啊! 概述 建造者模式将客户端与包含多个组成部分的复杂对象的创建过程分离,客户端压根不用知道复杂对象的内部组成部分与装配方式,只需要知道所需建造者的类型即可.它关注如何一步一步创建一个的复杂对象,不同的具体建造者定义了不同的创建过程,且具体建造者相互独立,增加新的建造者非常方便,无须修改已有代码,系统具有较好的扩展性. 问题来了... 你可能会有疑惑,建造者模式和抽象工

生成器模式(Builder)-- 对象创建型模式

1. 动机 将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示.一个RTF(Rich Text Format)文档交换格式的阅读器应能将RTF转换为多种正文格式.该阅读器可以将RTF文档转换成普通ASCII文本或转换成一个能以交互方式编辑的正文窗口组件.但问题在于可能转换的数目是无限的.因此要能够很容易实现新的转换的增加,同时却不改变RTF阅读器.其实也就是,前面的数据接卸(源头处理)归解析,后续的显示处理,由显示处理的部分来完成.在数据解析和显示处理之间架设一个标准的桥梁

Java设计模式——单例模式(创建型模式)

概述 ??单例模式保证对于每一个类加载器,一个类仅有一个实例并且提供全局的访问.其是一种对象创建型模式.对于单例模式主要适用以下几个场景: 系统只需要一个实例对象,如提供一个唯一的序列号生成器 客户调用类的单个实例只允许使用一个公共访问点,除了该公共访问点,不能通过其他途径访问该实例 ??单例模式的缺点之一是在分布式环境中,如果因为单例模式而产生 bugs,那么很难通过调试找出问题所在,因为在单个类加载器下进行调试,并不会出现问题. 实现方式 ??一般来说,实现枚举有五种方式:饿汉式.懒汉式.双

创建型-抽象工厂模式学习

1.抽象工厂模式的意图: 提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类. 2.抽象工厂模式的适用性: 一个系统要独立于它的产品的创建.组合和表示时. 一个系统要由多个产品系列中的一个来配置时. 当你要强调一系列相关的产品对象的设计以便进行联合使用时. 当你提供一个产品类库,而只想显示它们的接口而不是实现时. 3.场景描述: 考虑一个生产多种不同风格的家具的工厂(FurnitureFactory),不同风格的家具系列可以提供不同的门.窗.地板等的组合,为同一所住房可以提供不同

创建型-工厂方法模式

1.工厂方法模式意图: 定义一个用于创建对象的接口,让子类决定实例化哪一个类.Factory Method使一个类的实例化延迟到其子类. 2.应用场景: 考虑一个榨汁机的应用场景.榨汁机可以用来制作苹果汁.西瓜汁.橘汁等,即榨汁机可以用来制作水果汁.但是,夏天的冷饮店中,为了提高制作水果汁的效率和防止味道互串等情况,可能需要分别准备三个水果榨汁机,分别用来生产苹果汁.西瓜汁.橘汁等. 3.工厂方法模式类图: 角色: Product:定义工厂方法所创建对象.如场景描述中的果汁. ConcreteP

ProtoType(原型)-对象创建型模式

1.意图 用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象. 2.动机 通过拷贝或者"克隆"一个类的实例来创建新的实例. 3.适用性 当一个系统应该独立于它的产品创建.构成和表示时,要使用Prototype模式:以及 当要实例化的类是在运行时刻指定时,例如,通过动态装载:或者 为了避免创建一个与产品类层次平行的工厂类层次时或者 当一个类的实例只能有几个不同状态组合中的一种时.建立相应数目的原型并克隆它们,可能比每次用合适的状态手工实例化该类更方便一些. 4.结构 5.代码