2021年5月27日
概念性的东西理解一下即可,主要需要掌握浅拷贝、深拷贝的原理,以及代码实现。
原型设计模式(Prototype)是一种对象创建型模式,使用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象,主要用于创建重复的对象,同时又能保证性能。
工作原理是将一个原型对象传给那个要发动创建的对象,这个要发动创建的对象通过请求原型对象拷贝自己,来实现创建过程。
应该是最简单的设计模式了,实现一个接口,重写一个方法即完成了原型模式。
Prototype
:声明克隆方法的接口,是所有具体原型类的公共父类,Cloneable
接口。ConcretePrototype
:具体原型类。Client
:让一个原型对象克隆自身从而创建一个新的对象。Cloneable
接口public class Person implements Cloneable {
private String name;
private int age;
private List<String> list = new ArrayList<>();
// getter、setter 方法
public Person() {
System.out.println("构造函数调用");
}
// 通过覆盖 Object 的 clone() 方法可以实现浅拷贝
@Override
protected Person clone() throws CloneNotSupportedException {
return (Person) super.clone();
}
}
为什么说是浅拷贝呢?看下面client使用的例子
public class Client {
public static void main(String [] args) {
Person person1 = new Person();
person1.setAge(20);
person1.setName("dawson");
person1.getList().add("1v4");
// 浅拷贝
Person person2 = person1.clone();
person2.setName("rose");
person2.getList().add("v5");
}
}
将person1拷贝一份person2出来,person2修改了list,会导致person1的list也修改了。同时也可以发现person1和person2的list都指向同一个内存地址{ArrayList@491}。
对于浅拷贝(浅克隆):
Object
类的clone()
方法可以实现浅克隆。‘Serializable
接口public class Person implements Cloneable, Serializable {
private String name;
private int age;
private List<String> list = new ArrayList<>();
public Person() {
System.out.println("构造函数调用");
}
// 浅拷贝:通过覆盖 Object 的 clone() 方法可以实现浅拷贝
@Override
protected Person clone() throws CloneNotSupportedException {
return (Person) super.clone();
}
// 深拷贝:通过序列化 Serializable 方式来实现深拷贝
public Object deepClone() {
try {
// 输出:序列化
ByteArrayOutputStream baos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(baos);
oos.writeObject(this);
// 输入:反序列化
ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray());
ObjectInputStream ois = new ObjectInputStream(bais);
Person copyObj = (Person) ois.readObject();
return copyObj;
} catch (Exception e) {
e.printStackTrace();
return null;
}
}
}
client使用:
public class Client {
public static void main(String [] args) throws CloneNotSupportedException {
Person person1 = new Person();
person1.setAge(20);
person1.setName("dawson");
person1.getList().add("1v4");
// 深拷贝
Person person2 = (Person) person1.deepClone();
person2.setName("rose");
person2.getList().add("v5");
}
}
可以看出,person2的修改不影响person1,是真正的克隆了一份,list的内存地址也发生了改变,这就是深拷贝。
对于深拷贝(深克隆):
Serializable
等方式来实现。(序列化和反序列化,会使得内存地址改变。)优点:
缺点:
原文:https://www.cnblogs.com/ddhhdd/p/14820278.html