创建型模式:关注对象的创建过程,它描述如何将对象的创建、使用分离,让用户无需关心对象的创建细节,从而降低系统的耦合度,让设计方案易于修改、扩展。
一、概述
原型模式(Prototype Pattern):使用原型实例指定待创建对象的类型,并且通过复制这个原型来创建新的对象。
原型模式是一种特殊的创建型模式,它通过复制一个已有对象来获取更多的相同 / 相似的对象。
二、结构
- Prototype(抽象原型类):
声明克隆方法的接口,所有具体原型类的公共父类,可以是接口、抽象类、具体类。 - ConcretePrototype(具体原型类):
实现原型中的克隆方法,返回自己的一个克隆对象。
三、拷贝问题
1、浅拷贝
在浅拷贝中,原型对象的成员变量有两种情况:
- 值类型:复制一份给克隆对象
- 引用类型:复制引用给克隆对象(不同引用指向同一类型同一对象)
2、深拷贝
在深拷贝中,原型对象的成员变量有两种情况:
- 值类型:复制一份给克隆对象
- 引用类型:复制引用对象给克隆对象(不同引用指向同一类型不同对象)
四、举例实现
定义原型对象Person,内部有属性 Car 对象。
1、浅拷贝
实例代码:
public class Car {
private Integer id;
private String brand;
private String color;
private Double price;
/*
* 省略所有属性的 Setter、Getter 方法,重写 toString() 方法,无参、全参 构造器
*/
}
public class Person implements Cloneable {
private Integer id;
private String name;
private Car car;
/*
* 省略所有属性的 Setter、Getter 方法,重写 toString() 方法,无参、全参 构造器
*/
@Override
public Person clone() throws CloneNotSupportedException {
Object obj = null;
try {
obj = super.clone();
return (Person) obj;
} catch (CloneNotSupportedException e) {
System.out.println("不支持拷贝!");
return null;
}
}
}
测试代码:
Car car = new Car(1,"Benz","红色",200000.00);
Person p = new Person(1,"张三",car);
Person p2 = p.clone();
System.out.println("是否为同一引用:" + (p == p2));
System.out.println("是否为同一种类型:" + p.equals(p2));
p.getCar().setColor("蓝色");
System.out.println(p);
System.out.println(p2);
2、深拷贝
实例代码:
public class Car implements Serializable {
private Integer id;
private String brand;
private String color;
private Double price;
/*
* 省略所有属性的 Setter、Getter 方法,重写 toString() 方法,无参、全参 构造器
*/
}
public class Person implements Serializable {
private Integer id;
private String name;
private Car car;
/*
* 省略所有属性的 Setter、Getter 方法,重写 toString() 方法,无参、全参 构造器
*/
public Person deepClone() throws IOException, ClassNotFoundException {
//将对象写入流
ByteArrayOutputStream bao = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(bao);
oos.writeObject(this);
//将对象从流中取出
ByteArrayInputStream bis = new ByteArrayInputStream(bao.toByteArray());
ObjectInputStream ois = new ObjectInputStream(bis);
return (Person) ois.readObject();
}
}
测试代码:
Car car = new Car(1,"Benz","红色",200000.00);
Person p = new Person(1,"张三",car);
Person p2 = p.deepClone();
System.out.println("是否为同一引用:" + (p == p2));
System.out.println("是否为同一种类型:" + p.equals(p2));
p.getCar().setColor("蓝色");
System.out.println(p);
System.out.println(p2);
五、原型管理器
原型管理器(Prototype Manager) 将多个原型对象存储在一个集合中,以供拷贝,便于扩展。
六、特点
☯ 优点
- 提高已有实例对象的创建效率
- 扩展性较好,可以将具体原型写入配置文件
- 提供了简化的创建结构,拥有原型可以无需额外的工厂类
- 可以使用深拷贝保存对象的状态,支持将对象恢复到某一历史状态
☯ 缺点
- 需要为每个类配备一个克隆方法,违背了开闭原则
- 在深拷贝须编写较为复杂的代码