“不好意思,我是卧底!哇哈哈哈~”额......自从写了上一篇的观察者模式,就一直沉浸在这个角色当中,无法自拨。昨晚在看《使徒行者2》,有一集说到啊炮仗哥印钞票,我去,这就是想印多少就印多少的节奏。

但是我觉得他们印钞票的方法太low了,就用那“哧咔,哧咔~”的老机器没日没夜的印,看着都着急。

这里我们可以用原型模式优化印钞票的致富之路,为什么,继续往下看......

一、原型模式

定义

  用原型实例指定所有创建对象的类型,并且通过复制这个拷贝创建新的对象。

特点

  1)必须存在一个现有的对象,也就是原型实例,通过原型实例创建新对象。

  2)在Java中,实现Cloneable,并且因为所有的类都继承Object类重写clone()方法来实现拷贝。

使用场景

  • 大量的对象,并且类初始化时消耗的资源多。没人会嫌钱多的吧,除了某云。

  • 这些钞票的信息属性基本一致,可以调整个别的属性。

  • 印钞票的工序非常复杂,需要进行繁琐的数据处理。

UML图

从上面的UML图可以看出,原型模式涉及到的角色有如下三个:

  - 客户端角色:负责创建对象的请求。

  - 抽象原型角色:该角色是一个抽象类或者是接口,提供拷贝的方法。

  - 具体原型角色:该角色是拷贝的对象,需要重写抽象原型的拷贝方法,实现浅拷贝或者深拷贝。

二、实战

一起来印钞票,钞票实例必须实现Cloneable接口,该接口只充当一个标记,然后重写clone方法,具体原型角色代码如下:

public class Money implements Cloneable {private int faceValue;private Area area;public int getFaceValue() {return faceValue;}public void setFaceValue(int faceValue) {this.faceValue = faceValue;}public Money(int faceValue, Area area) {this.faceValue = faceValue;this.area = area;}public Area getArea() {return area;}public void setArea(Area area) {this.area = area;}public String getUnit() {return unit;}public void setUnit(String unit) {this.unit = unit;}@Overrideprotected Money clone() throws CloneNotSupportedException {return (Money) super.clone();}
}

Area类代码如下:

public class Area {// 钞票单位private String unit;public String getUnit() {return unit;}public void setUnit(String unit) {this.unit = unit;}}

看看客户端如何实现钞票的拷贝,代码如下:

public class Client {public static void main(String[] args) {Area area = new Area();area.setUnit("RMB");// 原型实例,100RMB的钞票Money money = new Money(100, area);for (int i = 1; i <= 3; i++) {try {Money cloneMoney = money.clone();cloneMoney.setFaceValue(i * 100);System.out.println("这张是" + cloneMoney.getFaceValue() +  cloneMoney.getArea().getUnit() + "的钞票");} catch (CloneNotSupportedException e) {e.printStackTrace();}}}
}

大把大把的钞票出来了

这张是100RMB的钞票

这张是200RMB的钞票

这张是300RMB的钞票

从上面并没有看到抽象原型角色的代码,那该角色在哪?Object就是这个抽象原型角色,因为Java中所有的类都默认继承Objet,在这提供clone方法。

三、浅拷贝和深拷贝

在使用原型模式的时候,常常需要注意用的到底是浅拷贝还是深拷贝,当然这必须结合实际的项目需求。下面来了解学习这两种拷贝的用法和区别:

首先我们来看一个例子,只改变客户端代码:

public class Client {public static void main(String[] args) {Area area = new Area();area.setUnit("RMB");// 原型实例,100RMB的钞票Money money = new Money(100, area);try {Money cloneMoney = money.clone();cloneMoney.setFaceValue(200);area.setUnit("美元"); System.out.println("原型实例的面值:" + money.getFaceValue() +money.getArea().getUnit());System.out.println("拷贝实例的面值:" + cloneMoney.getFaceValue() + cloneMoney.getArea().getUnit());} catch (CloneNotSupportedException e) {e.printStackTrace();}}}

运行结果如下:

原型实例的面值:100美元

拷贝实例的面值:200美元

浅拷贝

见鬼了,明明就把原型实例的单位改成了美元而已,拷贝实例怎么也会跟着改变的。哪里有鬼?其实是Java在搞鬼。我们用的是Object的clone方法,而该方法只拷贝按值传递的数据,比如String类型和基本类型,但对象内的数组、引用对象都不拷贝,也就是说内存中原型实例和拷贝实例指向同一个引用对象的地址,这就是浅拷贝。浅拷贝的内存变化如下图:

从上图可以看出,浅拷贝前后的两个实例对象共同指向同一个内存地址,即它们共有拥有area1实例,同时也存在着数据被修改的风险。注意,这里不可拷贝的引用对象是指可变的类成员变量

深拷贝

同样的看例子,客户端代码如下:

public class Client {public static void main(String[] args) {Area area = new Area();area.setUnit("RMB");// 原型实例,100RMB的钞票Money money = new Money(100, area);try {Money cloneMoney = money.clone();cloneMoney.setFaceValue(200);area.setUnit("美元");System.out.println("原型实例的面值:" + money.getFaceValue() + money.getArea().getUnit());System.out.println("拷贝实例的面值:" + cloneMoney.getFaceValue() + cloneMoney.getArea().getUnit());} catch (CloneNotSupportedException e) {e.printStackTrace();}}}

运行结果如下:

原型实例的面值:100美元

拷贝实例的面值:200RMB

咦~这客户端代码不是跟浅拷贝的一样吗,但是运行结果却又不一样了。关键就在,实现深拷贝就需要完全的拷贝,包括引用对象,数组的拷贝。所以Area类也实现了Cloneable接口,重写了clone方法,代码如下:

public class Area implements Cloneable{// 钞票单位private String unit;public String getUnit() {return unit;}public void setUnit(String unit) {this.unit = unit;}@Overrideprotected Area clone() throws CloneNotSupportedException {Area cloneArea;cloneArea = (Area) super.clone();return cloneArea;}
}

另外,在Money钞票类的clone方法增加拷贝Area的代码:

public class Money implements Cloneable, Serializable {private int faceValue;private Area area;public int getFaceValue() {return faceValue;}public void setFaceValue(int faceValue) {this.faceValue = faceValue;}public Money(int faceValue, Area area) {this.faceValue = faceValue;this.area = area;}public Area getArea() {return area;}public void setArea(Area area) {this.area = area;}@Overrideprotected Money clone() throws CloneNotSupportedException {Money cloneMoney = (Money) super.clone();cloneMoney.area = this.area.clone();  // 增加Area的拷贝return cloneMoney;}}

深拷贝的内存变化如下图:

深拷贝除了需要拷贝值传递的数据,还需要拷贝引用对象、数组,即把所有引用的对象都拷贝。需要注意的是拷贝的引用对象是否还有可变的类成员对象,如果有就继续对该成员对象进行拷贝,如此类推。所以使用深拷贝是注意分析拷贝有多深,以免影响性能。

序列化实现深拷贝

这是实现深拷贝的另一种方式,通过二进制流操作对象,从而达到深拷贝的效果。把对象写到流里的过程是序列化过程,而把对象从流中读出来的过程则叫反序列化过程。深拷贝的过程就是把对象序列化(写成二进制流),然后再反序列化(从流里读出来)。注意,在Java中,常常可以先使对象实现Serializable接口,包括引用对象也要实现Serializable接口,不然会抛NotSerializableException。

只要修改Money,代码如下:

public class Money implements Serializable {private int faceValue;private Area area;public int getFaceValue() {return faceValue;}public void setFaceValue(int faceValue) {this.faceValue = faceValue;}public Money(int faceValue, Area area) {this.faceValue = faceValue;this.area = area;}public Area getArea() {return area;}public void setArea(Area area) {this.area = area;}@Overrideprotected Money clone() throws CloneNotSupportedException {Money money = null;try {// 调用deepClone,而不是Object的clone方法cloneMoney = (Money) deepClone();} catch (IOException e) {e.printStackTrace();} catch (ClassNotFoundException e) {e.printStackTrace();}return cloneMoney;}// 通过序列化深拷贝public Object deepClone() throws IOException, ClassNotFoundException {//将对象写到流里ByteArrayOutputStream bos = new ByteArrayOutputStream();ObjectOutputStream oos = new ObjectOutputStream(bos);oos.writeObject(this);//从流里读回来ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());ObjectInputStream ois = new ObjectInputStream(bis);return ois.readObject();}
}

同样运行客户端代码,最后来看看结果:

原型实例的面值:100美元

拷贝实例的面值:200RMB

四、原型模式的优缺点

优点

1)提高性能。不用new对象,消耗的资源少。

缺点

1)浅拷贝时需要实现Cloneable接口,深拷贝则要特别留意是否有引用对象的拷贝。

总结

原型模式本身比较简单,重写Object的clone方法,实现浅拷贝还是深拷贝。重点在理解浅拷贝和深拷贝,这是比较细但又重要,却往往被忽略的知识点。好啦,原型模式就到这了,下一篇是策略模式,敬请关注,拜拜!

设计模式Java源码GitHub下载https://github.com/jetLee92/DesignPattern

转载于:https://www.cnblogs.com/AndroidJet/p/7737658.html

我的Java设计模式-原型模式相关推荐

  1. Java设计模式-原型模式

    原型模式   在有些系统中,存在大量相同或相似对象的创建问题,如果用传统的构造函数来创建对象,会比较复杂且耗时耗资源,用原型模式生成对象就很高效. 原型模式的定义与特点   原型(Prototype) ...

  2. Java设计模式——原型模式

    概述 原型模式是为了解决一些不必要的对象创建过程.当Java JDK中提供了Cloneable接口之后,原型模式就变得异常的简单了.虽然由于Cloneable的引入使用程序变得更简单了,不过还是有一些 ...

  3. 7.Java设计模式-----原型模式(Prototype Pattern)

    什么是原型模式? 原型模式(Prototype Pattern)是用于创建重复的对象,同时又能保证性能.这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式. 这种模式是实现了一个原型接口 ...

  4. Java设计模式--原型模式

    1 Prototype Pattern 原型模式 目的:在运行期通过"复制和粘贴"来创建新对象 : 实现:创建一个原型对象,再通过Java 语言中常用的克隆实现方法复制这个原型对象 ...

  5. java设计模式---原型模式

    在阎宏博士的<JAVA与模式>一书中开头是这样描述原型(Prototype)模式的: 原型模式属于对象的创建模式.通过给出一个原型对象来指明所有创建的对象的类型,然后用复制这个原型对象的办 ...

  6. Java设计模式--原型模式Prototype

    原型模式Prototype 原型模式使得用户可以通过复制对象样本来创建新对象.与通过调用构造函数创建对象相比,二者主要区别在于:通过复制创建的新对象一般会包含原始对象的某些状态. 原型模式属于对象的创 ...

  7. Java常用设计模式————原型模式(一)

    介绍 原型模式(Prototype Pattern):用原型实例指定创建对象的种类,并通过拷贝这些原型创建新的对象. 原型模式用于创建重复的对象,同时又能保证性能.当直接创建对象的代价比较大时,则采用 ...

  8. 设计模式 原型模式_创新设计模式:原型模式

    设计模式 原型模式 原型模式用于创建对象的副本. 这种模式非常有用,特别是当从头开始创建对象的成本很高时. 与builder , factory和abstract factory模式相比,它不会从头开 ...

  9. 设计模式 原型模式_设计模式:原型

    设计模式 原型模式 创新设计模式之一是原型设计模式 . 尽管原型是创造模式,但它在概念上与其他模式有所区别. 我的意思是原型在某种意义上创造了自己. 我将在下面解释. 原型模式的所有魔力都基于Java ...

最新文章

  1. 51Nod:1085 背包问题
  2. RHCE课程-RH253Linux服务器架设笔记五-APACHE服务器配置(4)
  3. Cordova探险系列(一个)
  4. ERP选型技巧之“三不要一要”
  5. python3 实现 A+B Problem(百练OJ:1000)
  6. python图片转文字_【收藏】图片转成文字的方法总结,python批量图片转文字信息参考源码...
  7. 第一个smarty例子--分页显示数据
  8. python中type(12.34)_下面代码的输出结果是
  9. system_Class类说明文档
  10. (转)Windows 批处理(bat)语法大全
  11. 02组团队项目-Alpha冲刺-3/6
  12. java解方程_JAVA解N元一次方程组(矩阵) | 学步园
  13. git提交代码的时候未获取最新代码
  14. 什么是南向接口和北向接口,南北流量和东西流量
  15. 修改win服务器防火墙端口号,Windows Server 修改防火墙和远程桌面(3389)默认端口...
  16. 视觉学习笔记Week9 Gazebo仿真环境下相机焦距计算
  17. 太原师范学院计算机考研资料汇总
  18. NUC972 使用不同的FLASH开机的ENV文件写法 NAND...
  19. pikachu之xss漏洞学习
  20. 创业这10种死法,你死在哪一种了

热门文章

  1. LR录制脚本中文乱码问题
  2. python判断字符串是纯数字_python判断字符串是否纯数字的方法
  3. append有时加载不出来_关于艾拉浏览器看漫画,有时加载慢的解决方法,你知道了吗...
  4. 子网掩码及网络号路由表的匹配方法
  5. linux 清理内存的c函数,Linux C函数之内存配置函数
  6. PHP类参数_PHP依赖注入的含义
  7. 搭建linux软件仓库,创建自己的YUM仓库
  8. json反射java对象_Jackson通过反射将Json转化为java对象
  9. 如何用 J-Link 来串口调试?
  10. 又一国产开源微内核操作系统上线!源代码已开放下载