(十六)原型模式建模与实现
最后更新于:2022-04-01 09:31:33
原型模式(Prototype):该模式的思想就是将一个对象作为原型,对其进行复制、克隆,产生一个和原对象类似的新对象。而这里的复制有两种:浅复制、深复制。
**浅复制**:将一个对象复制后,基本数据类型的变量都会重新创建,而引用类型,指向的还是原对象所指向的。
**深复制**:将一个对象复制后,不论是基本数据类型还有引用类型,都是重新创建的。简单来说,就是深复制进行了完全彻底的复制,而浅复制不彻底。
下面通过示例进行说明:
##一、浅复制
### 1、uml建模:
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-02-19_56c7196e8ff4c.jpg)
### 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建模:
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-02-19_56c7196ea9285.jpg)
### 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、要实现深复制,需要采用二进制流的形式写入当前对象,再对其进行读取。