不管怎么样,都要继续充满着希望
上一章简单介绍了抽象工厂模式(六), 如果没有看过,请观看上一章
一. 原型模式
引用 菜鸟教程里面的原型模式介绍: https://www.runoob.com/design-pattern/prototype-pattern.html
原型模式(Prototype Pattern)是用于创建重复的对象,同时又能保证性能。这种类型的设计模式属于创建型模式,
它提供了一种创建对象的最佳方式之一。
这种模式是实现了一个原型接口,该接口用于创建当前对象的克隆。当直接创建对象的代价比较大时,则采用这种模式。
例如,一个对象需要在一个高代价的数据库操作之后被创建。我们可以缓存该对象,在下一个请求时返回它的克隆,
在需要的时候更新数据库,以此来减少数据库调用
一.一 介绍
意图:用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。
主要解决:在运行期建立和删除原型。
何时使用: 1、当一个系统应该独立于它的产品创建,构成和表示时。 2、当要实例化的类是在运行时刻指定时,例如,通过动态装载。 3、为了避免创建一个与产品类层次平行的工厂类层次时。 4、当一个类的实例只能有几个不同状态组合中的一种时。建立相应数目的原型并克隆它们可能比每次用合适的状态手工实例化该类更方便一些。
如何解决:利用已有的一个原型对象,快速地生成和原型对象一样的实例。
关键代码: 1、实现克隆操作,在 JAVA 实现 Cloneable 接口,重写 clone(),在 .NET 中可以使用 Object 类的 MemberwiseClone() 方法来实现对象的浅拷贝或通过序列化的方式来实现深拷贝。 2、原型模式同样用于隔离类对象的使用者和具体类型(易变类)之间的耦合关系,它同样要求这些"易变类"拥有稳定的接口。
应用实例: 1、细胞分裂。 2、JAVA 中的 Object clone() 方法。
优点: 1、性能提高。 2、逃避构造函数的约束。
缺点: 1、配备克隆方法需要对类的功能进行通盘考虑,这对于全新的类不是很难,但对于已有的类不一定很容易,特别当一个类引用不支持串行化的间接对象,或者引用含有循环结构的时候。 2、必须实现 Cloneable 接口。
使用场景:
1、资源优化场景。
2、类初始化需要消化非常多的资源,这个资源包括数据、硬件资源等。
3、性能和安全要求的场景。
4、通过 new 产生一个对象需要非常繁琐的数据准备或访问权限,则可以使用原型模式。
5、一个对象多个修改者的场景。
6、一个对象需要提供给其他对象访问,而且各个调用者可能都需要修改其值时,
可以考虑使用原型模式拷贝多个对象供调用者使用。
7、在实际项目中,原型模式很少单独出现,一般是和工厂方法模式一起出现,通过 clone 的方法创建一个对象,
然后由工厂方法提供给调用者。原型模式已经与 Java 融为浑然一体,大家可以随手拿来使用。
注意事项:与通过对一个类进行实例化来构造新对象不同的是,原型模式是通过拷贝一个现有对象生成新对象的。浅拷贝实现 Cloneable,重写,深拷贝是通过实现 Serializable 读取二进制流。
二. 原型模式实现
定义一个对象 Sheep
@Data
public class Sheep {
private Integer id;
private String name;
}
二.一 未使用原型模式
测试方法:
@Test
public void oneTest (){
Sheep sheep = new Sheep();
sheep.setId(1);
sheep.setName("羊");
Sheep newSheep = sheep;
log.info("新的 {}", newSheep);
newSheep.setName("名称改变了:");
log.info("改变后的名称: {}", newSheep);
log.info("之前老的数据: {}", sheep);
}
会将原先的老对象也改变。
重新构建对象, new Sheep() 的话
@Test
public void twoTest (){
Sheep sheep = new Sheep();
sheep.setId(1);
sheep.setName("羊");
Sheep newSheep = new Sheep();
newSheep.setId(sheep.getId());
newSheep.setName(sheep.getName());
log.info("新的 {}", newSheep);
newSheep.setName("名称改变了:");
log.info("改变后的名称: {}", newSheep);
log.info("之前老的数据: {}", sheep);
}
不会改变老的, 但每次都是新 new 对象, 需要将每一个属性都放置进去, 太复杂.
二.二 实现 Cloneable 接口,复制对象
@Data
public class ThreeSheep implements Cloneable{
private Integer id;
private String name;
@Override
public Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
测试方法:
@Test
public void threeTest () throws Exception{
ThreeSheep sheep = new ThreeSheep();
sheep.setId(1);
sheep.setName("羊");
ThreeSheep newSheep = (ThreeSheep) sheep.clone();
log.info("新的 {}", newSheep);
newSheep.setName("名称改变了:");
log.info("改变后的名称: {}", newSheep);
log.info("之前老的数据: {}", sheep);
}
通过 .clone() 方法 将 原先的对象进行复制, 这就是原型模式.
二.三 深复制,浅复制
定义一个 User 对象 和 Dept 对象, User 里面依赖 Dept 对象
二.三.一 浅复制
User
@Data
public class User implements Cloneable {
private Integer id;
private String name;
private Dept dept;
@Override
public Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
Dept.java
@Data
public class Dept implements Cloneable{
private Integer id;
private String name;
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
测试方法:
@Test
public void fourTest() throws Exception {
User user = new User();
user.setId(1);
user.setName("张三");
Dept dept = new Dept();
dept.setId(1);
dept.setName("研发部");
user.setDept(dept);
log.info("构建用户: {}",user);
User cloneUser = (User) user.clone();
cloneUser.setName("张三Copy");
log.info("克隆后的用户: {}",cloneUser);
log.info("之前的用户: {}",user);
// 会发现 , Dept 同时改变了, 即 Dept 并没有 Copy 成功。
cloneUser.getDept().setName("信息部");
log.info("修改后的用户: {}",cloneUser);
log.info("之前的用户: {}",user);
}
可以发现,会将 Dept 引用的对象 也进行修改了。
浅复制即 只复制了当前对象的普通属性, 并没有将引用属性也进行复制。
二.三.二 深复制
@Data
public class NewDept implements Cloneable{
private Integer id;
private String name;
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
注意 clone() 方法内的写法, 先进行 dept 的 clone() 再设置到 dept 属性里面。
@Data
public class NewUser implements Cloneable {
private Integer id;
private String name;
private NewDept dept;
@Override
public Object clone() throws CloneNotSupportedException {
NewUser newUser = (NewUser) super.clone();
NewDept deptCopy = (NewDept) dept.clone();
newUser.setDept(deptCopy);
return newUser;
}
}
测试方法:
@Test
public void fiveTest() throws Exception {
NewUser user = new NewUser();
user.setId(1);
user.setName("张三");
NewDept dept = new NewDept();
dept.setId(1);
dept.setName("研发部");
user.setDept(dept);
log.info("构建用户: {}",user);
NewUser cloneUser = (NewUser) user.clone();
cloneUser.setName("张三Copy");
log.info("克隆后的用户: {}",cloneUser);
log.info("之前的用户: {}",user);
cloneUser.getDept().setName("信息部");
log.info("修改后的用户: {}",cloneUser);
log.info("之前的用户: {}",user);
}
可以发现, dept 对象并没有改变, user.clone() 方法 将 user 普通属性 和 dept 对象都进行了复制。
二.四 序列化实现深复制
需要实现 Serializable 接口
@Data
public class NewDept2 implements Serializable{
private Integer id;
private String name;
}
@Data
public class NewUser2 implements Serializable {
private Integer id;
private String name;
private NewDept2 dept;
/**
深复制
*/
public Object deepClone() {
// 创建流对象
ByteArrayOutputStream bos = null;
ObjectOutputStream oos = null;
ByteArrayInputStream bis = null;
ObjectInputStream ois = null;
try {
// 序列化
bos = new ByteArrayOutputStream();
oos = new ObjectOutputStream(bos);
oos.writeObject(this);
// 反序列化
bis = new ByteArrayInputStream(bos.toByteArray());
ois = new ObjectInputStream(bis);
NewUser2 copyObj = (NewUser2)ois.readObject();
return copyObj;
}catch (Exception e) {
e.printStackTrace();
return null;
}finally {
try {
bos.close();
oos.close();
bis.close();
ois.close();
}catch (Exception e) {
}
}
}
}
测试方法:
@Test
public void sexTest(){
NewUser2 user = new NewUser2();
user.setId(1);
user.setName("张三");
NewDept2 dept = new NewDept2();
dept.setId(1);
dept.setName("研发部");
user.setDept(dept);
log.info("构建用户: {}",user);
NewUser2 cloneUser = (NewUser2) user.deepClone();
cloneUser.setName("张三Copy");
log.info("克隆后的用户: {}",cloneUser);
log.info("之前的用户: {}",user);
cloneUser.getDept().setName("信息部");
log.info("修改后的用户: {}",cloneUser);
log.info("之前的用户: {}",user);
}
是实现了深复制的.
二.五 通过第三方工具类 实现深复制
如通过 hutool 的 ObjectUtil.clone 实现
NewUser2 和 NewDept2 与上面 二.四 的一样,都需要实现序列化接口
@Test
public void sevenTest() {
NewUser2 user = new NewUser2();
user.setId(1);
user.setName("张三");
NewDept2 dept = new NewDept2();
dept.setId(1);
dept.setName("研发部");
user.setDept(dept);
log.info("构建用户: {}",user);
// 通过第三方对象, 需要序列化接口。
NewUser2 cloneUser = ObjectUtil.clone(user);
cloneUser.setName("张三Copy");
log.info("克隆后的用户: {}",cloneUser);
log.info("之前的用户: {}",user);
cloneUser.getDept().setName("信息部");
log.info("修改后的用户: {}",cloneUser);
log.info("之前的用户: {}",user);
}
深复制也是成功的。
本章节的代码放置在 github 上:
https://github.com/yuejianli/DesignPattern/tree/develop/Prototype
谢谢您的观看,如果喜欢,请关注我,再次感谢 !!!