创建对象_原型(Prototype)模式_深拷贝
顾客:“给我几个快递。”快递员:“寄往什么地方?寄给...?”顾客:“和上次差不多一样,只是邮寄给另外一个地址,这里是邮寄地址...“把邮寄地址的纸条给快递员。快递员:“好。”
- 要实现克隆,必须实现 java.lang.Cloneable 接口,否则在运行时调用 clone( ) 方法,会抛出 CloneNotSupportedException异常。
- 返回的是 Object类型的对象,所以使用时可能需要强制类型转换。
- 该方法是 protected的,如果想让外部对象使用它,必须在子类重写该方法,设定其访问范围是 public的,参见 PackageInfo 的 clone( ) 方法。
- Object 的 clone( ) 方法的复制是采用逐字节的方式从内存赋值数据,复制了属性了引用,而属性所指向的对象本身没有被复制,因此所复制的引用指向了相同的对象。即 浅拷贝。
public class PackageInfo implements Cloneable {
public PackageInfo clone() {
try {
return (PackageInfo) super .clone();
} catch (CloneNotSupportedException e) {
System. out .println( "Cloning not allowed." );
return null ;
}
}
// 静态工厂方法:根据原型创建一份副本
public static PackageInfo clonePackage(String userName) {
// 根据 userName加载一条用户以前的数据作为原型数据(数据库或其它保存的数据)
PackageInfo prototype = loadPackageInfo (userName);
// 再在内存中克隆这条数据
prototype = prototypr .clone();
// 初始化数据 id(主键)
prototype. setId ( null );
// 返回数据
return prototype;
}
}
- 复制对象时,递归地调用属性对象的克隆方法。根据具体的类,撰写出实现特定类型的深拷贝方法。
一般我们很难实现一个一般性的方法来完成任何类型对象的深拷贝。根据反射得到属性的类型,然后依照它的类型构造对象,但前提是:这些属性的类型必须含有一个公有的默认构造方法,否则作为一个一般性的方法,很难确定传递给非默认构造方法的参数值;此外,如果属性类型是接口或者抽象类型,必须提供查找到相关的具体类方法,作为一个一般性的方法,这个也很难办到。
- 如果类实现了 java.io.Serializable 接口,把原型对象序列化,然后反序列化后得到得对象,其实就是一个新的深拷贝对象。
//DeepCopyBean实现了 java.io.Serializable接口
public class DeepCopyBean implements Serializable {
// 原始类型属性
private int primitiveField ;
// 对象属性
private String objectField ;
// 首先序列化自己到流中,然后从流中反序列化,得到得对象便是一个新的深拷贝
public DeepCopyBean deepCopy() {
try {
ByteArrayOutputStream buf = new ByteArrayOutputStream();
ObjectOutputStream o = new ObjectOutputStream(buf);
o.writeObject( this );
ObjectInputStream in = new ObjectInputStream(
new ByteArrayInputStream(buf.toByteArray()));
return (DeepCopyBean) in.readObject();
} catch (Exception e) {
e.printStackTrace();
}
return null ;
}
// 属性 get、set 方法略... // 测试demo
public static void main(String[] args) {
DeepCopyBean originalBean = new DeepCopyBean();
// 创建两个 String对象,其中一个在 JVM的字符串池(String pool)里,属性引用指向另外一个在堆里的对象
originalBean.setObjectField( new String( "guilin" ));
originalBean.setPrimitiveField(50);
// 深拷贝
DeepCopyBean newBean = originalBean.deepCopy();
// 原始类型属性值比较:true
System. out .println( "primitiveField ==:"
+ (originalBean.getPrimitiveField() == newBean
.getPrimitiveField()));
// 对象属性值比较:false(证明未指向相同的地址)
System. out .println( "objectField ==:"
+ (originalBean.getObjectField() == newBean.getObjectField()));
// 对象属性 equals 比较:true
System. out .println( "objectField equal:"
+ (originalBean.getObjectField().equals(newBean
.getObjectField())));
}
}
- 它只能复制实现 Serializable接口类型的对象,其属性也是可序列化的;
- 序列化和反序列化比较耗时。
- 创建大的聚合对象图时,没有必要为每个层次的子对象创建相应层次的工厂类。
- 方便实例化,只要复制对象,然后初始化对象,就可以得到想要的对象。
创建对象_原型(Prototype)模式_深拷贝的更多相关文章
- 桥接模式_NAT模式_仅主机模式_模型图.ziw
2017年1月12日, 星期四 桥接模式_NAT模式_仅主机模式_模型图 null
- 原型(Prototype)模式
原型模式属于对象的创建模式.通过给出一个原型对象来指明所有创建的对象的类型,然后用复制这个原型对象的办法创建出更多同类型的对象.这就是原型模式的用意.原型模式的结构 原型模式要求对象实现一个可以“克隆 ...
- Java 实现原型(Prototype)模式
public class BaseSpoon implements Cloneable {//spoon 匙, 调羹 String name; public String getName() { re ...
- 设计模式C++描述----08.原型(Prototype)模式
一. 概述 定义:用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象. 换句话说,就是不用重新初始化对象,而是动态地获得对象运行时的状态. 再说明白点,就是要一个拷贝过构造函数类似功能的接 ...
- 设计模式--原型(Prototype)模式
写这些也许有人认为“为了模式而模式”.Insus.NET所想到的,每个大师成为大师之前,也许都得这样做. 走路,从小就开始学,直至现在,谁还不是为了走路而走路?一直重复着...... 很多人没有分享自 ...
- 六、原型(Prototype)模式
原型模式是对象的创建模式,通过给出一个原型对象来指明所要创建的对象的类型.然后用复制这个原型对象的方法来创建出更多同类型的对象. 原型模式可以不用重新初始化对象,而动态的获取对象运行时的状态.使用原型 ...
- 分组密码_计数器(CTR)模式_原理及java实现
一.原理: CTR模式是一种通过将逐次累加的计数器进行加密来生成密钥流的流密码,在CTR模式中,每个分组对应一个逐次累加的计数器,并通过对计数器进行加密来生成密钥流.最终的密文分组是通过将计数器加密得 ...
- Java设计模式_创建型模式_单例模式
单例模式的实现: 定义一个类,在类中定义该类的静态变量,再定一个一个获取该类的静态变量的方法. UML图:
- 克隆复制可使用原型( Prototype)设计模式
今天有学习设计模式的原型(Prototype)<设计模式--原型(Prototype)模式>http://www.cnblogs.com/insus/p/4152773.html .为了加 ...
- C#设计模式:原型模式(Prototype)及深拷贝、浅拷贝
原型模式(Prototype) 定义: 原型模式:用原型实例指定创建对象的种类,并且通过复制这些原型创建新的对象.被复制的实例被称为原型,这个原型是可定制的. Prototype Pattern也是一 ...
随机推荐
- UVALive 7079 - How Many Maos Does the Guanxi Worth(最短路Floyd)
https://icpcarchive.ecs.baylor.edu/index.php?option=com_onlinejudge&Itemid=8&page=show_probl ...
- poj 1797 Heavy Transportation(Dijkstar变形)
http://poj.org/problem?id=1797 给定n个点,及m条边的最大负载,求顶点1到顶点n的最大载重量. 用Dijkstra算法解之,只是需要把“最短路”的定义稍微改变一下, A到 ...
- poj 3264 Balanced Lineup(RMQ裸题)
Balanced Lineup Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 43168 Accepted: 20276 ...
- 【破解三网】iphone5 国行 A1429
教程仅适用于A1429的I5. 1.首先把sim卡插入手机.关闭蜂窝数据里面的"蜂窝移动数据"还有打开漫游里面的"语音漫游".如图 ...
- HeapAlloc、GlobalAlloc和new等内存分配有什么区别么?
查找了一些 new , GlobalAlloc, HeapAlloc分配内存方式的区别. 转了一些资料 //============================================== ...
- android 简易定时器
定时器 1.在android 应用开发当中,很多时候都要用到定时器,而要实现定时器更多的时候要用到两个类:Timer,和TimerTask 2.API对Timer的解释是:
- js打开新页面 关闭当前页 关闭父页面
js打开新页面.关闭当前页.关闭父页面 2010-04-29 14:04:13| 分类: 页面与JavaScript | 标签: |字号大中小 订阅 //关闭当前页面,并且打开新页面,(不 ...
- ngTbale假分页实现排序、搜索、导出CSV等功能
一. ngTable功能简化 使用ngTable经常有分页,排序,过滤等功能,实现诸多功能较为麻烦.为了方便开发过程,可以抽取一些table共同点写一个公有方法. 注意: 1. 由于很多特别的需求,可 ...
- 以静态变量保存 Spring ApplicationContext
package com.thinkgem.jeesite.common.utils; import java.net.HttpURLConnection; import java.net.URL; i ...
- js ajax上传图片到服务器
$("#up_goods_pic").on('change',function(){ var file = this.files[0]; var url = webkitURL.c ...