Java设计模式菜鸟系列(十六)原型模式建模与实现
来源:互联网 发布:人工智能发展现状 编辑:程序博客网 时间:2024/05/22 06:26
转载请注明出处:http://blog.csdn.net/lhy_ycu/article/details/39997337
原型模式(Prototype):该模式的思想就是将一个对象作为原型,对其进行复制、克隆,产生一个和原对象类似的新对象。而这里的复制有两种:浅复制、深复制。
浅复制:将一个对象复制后,基本数据类型的变量都会重新创建,而引用类型,指向的还是原对象所指向的。
深复制:将一个对象复制后,不论是基本数据类型还有引用类型,都是重新创建的。简单来说,就是深复制进行了完全彻底的复制,而浅复制不彻底。
下面通过示例进行说明:
一、浅复制
1、uml建模:
2、代码实现
/** * 原型模式:将一个对象作为原型,对其进行复制、克隆,产生一个和原对象类似的【新对象】。 * * 而这里的复制有两种:浅复制、深复制 * * 示例(一) 浅复制:将一个对象复制后,基本数据类型的变量都会重新创建, * * 而引用类型,指向的还是原对象所指向的,【不会重新创建】。 */class Prototype implements Cloneable {private int age;private int[] array = new int[] { 1, 2, 3 };public Prototype() {}public Prototype(int age) {this.age = age;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public int[] getArray() {return array;}public void setArray(int[] array) {this.array = array;}/** * 因为Cloneable接口是个空接口 * * 此处的重点是super.clone()这句话,super.clone()调用的是Object的clone()方法,而在Object类中,clone()是native的 * * 这就涉及到JNI,关于JNI还有NDK以后会讲到,这里你只要记住浅复制的核心是super.clone()。 */public Object cloneObject() throws CloneNotSupportedException {Prototype prototype = (Prototype) super.clone();return prototype;}}/** * 客户端测试类 * * @author Leo */public class Test {public static void main(String[] args) throws CloneNotSupportedException {Prototype prototype = new Prototype(20);Prototype cloneProto = (Prototype) prototype.cloneObject();/** * 通过打印可以看到:prototype和cloneProto这两个同一类型的变量指向的是两个不同的内存地址 * * 这说明克隆成功 */System.out.println("prototype = " + prototype);System.out.println("cloneProto = " + cloneProto);/** * 要完全复制一个对象的话,那么它的引用类型变量array指向的肯定是不同的内存地址 * * 而这里的引用类型变量array,指向的还是原对象所指向的。可以看到打印的内存地址是相同的。 * * 这说明对象复制不彻底 */System.out.println("prototype.getArray() = " + prototype.getArray());System.out.println("cloneProto.getArray() = " + cloneProto.getArray());/** * 透过这个例子可以看到:浅复制并没有将对象进行完全复制 */}}
二、深复制
1、uml建模:
2、代码实现
/** * 示例(二) 深复制:将一个对象复制后,不论是基本数据类型还有引用类型,都是【重新创建】的。 * * 简单来说,就是深复制进行了完全彻底的复制,而浅复制不彻底。 * * 由于这里涉及到对对象的读写,所以这里用到了对象的序列化--实现了Serializable接口 */class Prototype implements Cloneable, Serializable {private static final long serialVersionUID = 1L;private int age;private int[] array = new int[] { 1, 2, 3 };public Prototype() {}public Prototype(int age) {this.age = age;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public int[] getArray() {return array;}public void setArray(int[] array) {this.array = array;}/* 深复制 */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();}}/** * 客户端测试类 * * @author Leo */public class Test {public static void main(String[] args) throws IOException,ClassNotFoundException {Prototype prototype = new Prototype(20);Prototype cloneProto = (Prototype) prototype.deepClone();/** * 通过打印可以看到:prototype和cloneProto这两个同一类型的变量指向的是两个不同的内存地址 * * 这说明克隆成功 */System.out.println("prototype = " + prototype);System.out.println("cloneProto = " + cloneProto);/** * 通过打印可以看到,两个对象的引用类型变量array指向的是不同的内存地址 * * 这说明对象进行了完全彻底的复制 */System.out.println("prototype.getArray() = " + prototype.getArray());System.out.println("cloneProto.getArray() = " + cloneProto.getArray());/** * 当然我们也可以试着打印一下引用变量的内容, * * 可以看到:内容是不变的(1 2 3),改变的只是引用变量指向的内存地址。 */int[] proArray = prototype.getArray();int[] cloneProtoArray = cloneProto.getArray();for (int p : proArray) {System.out.print(p + "\t");}System.out.println();for (int p : cloneProtoArray) {System.out.print(p + "\t");}}}
三、总结
1、浅复制的核心是super.clone(),它调用的是Object的clone()方法,而在Object类中,clone()是native的。
2、要实现深复制,需要采用二进制流的形式写入当前对象,再对其进行读取。
1 0
- Java设计模式菜鸟系列(十六)原型模式建模与实现
- Java设计模式菜鸟系列(一)策略模式建模与实现
- Java设计模式菜鸟系列(二)观察者模式建模与实现
- Java设计模式菜鸟系列(三)装饰者模式建模与实现
- Java设计模式菜鸟系列(四)工厂方法模式建模与实现
- Java设计模式菜鸟系列(五)抽象工厂模式建模与实现
- Java设计模式菜鸟系列(六)单例模式建模与实现
- Java设计模式菜鸟系列(七)命令模式建模与实现
- Java设计模式菜鸟系列(八)适配器模式建模与实现
- Java设计模式菜鸟系列(九)外观模式建模与实现
- Java设计模式菜鸟系列(十)模板方法模式建模与实现
- Java设计模式菜鸟系列(十一)迭代器模式建模与实现
- Java设计模式菜鸟系列(十二)组合模式建模与实现
- Java设计模式菜鸟系列(十三)状态模式建模与实现
- Java设计模式菜鸟系列(十四)代理模式建模与实现
- Java设计模式菜鸟系列(十五)建造者模式建模与实现
- Java设计模式菜鸟系列(十七)桥接模式建模与实现
- Java设计模式菜鸟系列(十八)责任链模式建模与实现
- Linux 安装FFmpeg安装配置
- c#中volatile关键字的作用
- 静态代码块、构造代码块以及构造函数的加载顺序
- linux服务器性能检测工具nmon使用
- Office word excel电子表格在线编辑的实现方法
- Java设计模式菜鸟系列(十六)原型模式建模与实现
- Android获取状态栏和标题栏的高度
- 通过Nginx搭建flv流媒体服务器
- linux下杀死进程(kill)的N种方法
- 约瑟夫环问题
- Dom解析xml文件的添加与删除方法
- log4j--控制台/文本/数据库---三模式
- 安装软件时对话框默认允许一路安装
- mysql 之 row_number() over ()