3、原形模式
-
听名字就知道,原形模式是提供一个模板,供其他实例进行参考(拷贝)
-
被拷贝的对象需要实现java.lang.Cloneable接口,然后重写Object类中的方法。这时候你要是认真的话你就会想了,尼玛,实现Cloneable接口,却去重写Object中的方法,有没有搞错?hhh,其实Cloneable是一个“标记接口”这个接口中没有任何内容,只是赋予了实现这个接口的类一个标志。只有实现这个接口的类才能重写Object类中的clone方法。否则会抛出异常。
- 在了解原形模式之前我们先了解一下什么是“深拷贝”和“浅拷贝”
- 浅拷贝:
- 当类的成员变量是基本数据类型时,浅拷贝会复制该属性的值赋值给新对象。
- 当成员变量是引用数据类型时,浅拷贝复制的是引用数据类型的地址值。这种情况下,当拷贝出的某一个类修改了引用数据类型的成员变量后,会导致所有拷贝出的类都发生改变。
- 深拷贝:
- 深拷贝不仅会复制成员变量为基本数据类型的值,给新对象。
- 还会给是引用数据类型的成员变量申请储存空间,并复制引用数据类型成员变量的对象。这样拷贝出的新对象就不怕修改了是引用数据类型的成员变量后,对其它拷贝出的对象造成影响了
-
package com.yunyanchengyu.designModel; public class DoLi { public static void main(String[] args) throws CloneNotSupportedException { Sheep sheep = new Sheep("mianyang","7"); //克隆过程中没有输出“我是原形”,说明克隆过程中没有调用构造函数 Sheep duoLi = sheep.clone(); //此条输出语句输出的是mianyang,可见复制成功了! System.out.println(duoLi.name); //给复制体修改名字(偷懒,没用getter/setter) //此时输出语句是duoLi,改名字成功 duoLi.name = "duoLi"; System.out.println(duoLi.name); //重点在这里,输出语句是mianyang。说明修改复制体属性并没有修改原形的属性。 //但是如果只是duoLi = sheep的话,修改duoLi的属性会影响sheep,大家可以去试一下 System.out.println(sheep.name); } } class Sheep implements Cloneable { String name; String age; public Sheep(String name, String age) { System.out.println("我是原形"); this.name = name; this.age = age; } //重写Object类的clone方法 @Override public Sheep clone() throws CloneNotSupportedException { return (Sheep) super.clone(); } }
-
总结,引用类型使用原形模式可以实现深拷贝。并且不会受构造函数和修饰符的影响。原型模式和单例模式是冲突的,大家可以思考下。