介绍
- 原型模式(Prototype模式)是指:用原型实例指定创建对象的种类,并且通过拷贝这些原型,创建新的对象
- 原型模式是一种创建型设计模式,允许一个对象再创建另外一个可定制的对象,无需知道如何创建的细节
- 工作原理是:通过将一个原型对象传给那个要发动创建的对象,这个要发动创建的对象通过请求原型对象拷贝它们自己来实施创建,即 对象.clone()
个人总结
如果需要多个相同的定制化对象时(除了对象内存地址不一样),并不需要多次以相同方法创建,这个操作复杂,只需要调用对象.clone()克隆一个这样的对象即可
浅拷贝
//一个person类实现Cloneable接口的clone方法
//person具有两个基本数据类型的属性name age和一个引用类型的属性address
public class Person implements Cloneable{
String name;
int age;
Address address;
public Person(String name, int age, Address address) {
this.name = name;
this.age = age;
this.address = address;
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
", address=" + address +
'}';
}
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
//Address类
public class Address {
String city;
String street;
int emailNum;
public Address(String city, String street, int emailNum) {
this.city = city;
this.street = street;
this.emailNum = emailNum;
}
}
//测试
public class test {
public static void main(String[] args) throws Exception {
Address addr = new Address("广州市", "青云街", 40033);
Person p1 = new Person("小明", 18, addr);
Person p2 = (Person)p1.clone();
System.out.println(p1);
System.out.println(p2);
System.out.println("p1对象---------->"+p1.hashCode());
System.out.println("p2对象---------->"+p2.hashCode());
System.out.println("p1的地址---------->"+p1.address.hashCode());
System.out.println("p2的地址---------->"+p2.address.hashCode());
}
}
测试输出控制台结果:
可以看出虽然使用p1为模板克隆出了一个p2,但是p1和p2的引用属性address并没有真正的克隆,他们只是指向了一个相同的address对象
深拷贝(重写clone方法)
//一个person类实现Cloneable接口的clone方法
//person具有两个基本数据类型的属性name age和一个引用类型的属性address
//实现深拷贝需要重写clone方法
public class Person implements Cloneable{
String name;
int age;
Address address;
public Person(String name, int age, Address address) {
this.name = name;
this.age = age;
this.address = address;
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
", address=" + address +
'}';
}
@Override
protected Object clone() throws CloneNotSupportedException {
Object obj = super.clone(); //首先把基本数据类型的属性克隆一次
Person person = (Person)obj;
person.address=(Address) address.clone(); //在把引用类型的基本数据类型复制一次
return person;
}
}
//Address类需要实现Cloneable的clone方法
public class Address implements Cloneable{
String city;
String street;
int emailNum;
public Address(String city, String street, int emailNum) {
this.city = city;
this.street = street;
this.emailNum = emailNum;
}
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
//测试
public class test {
public static void main(String[] args) throws Exception {
Address addr = new Address("广州市", "青云街", 40033);
Person p1 = new Person("小明", 18, addr);
Person p2 = (Person)p1.clone();
System.out.println(p1);
System.out.println(p2);
System.out.println("p1对象---------->"+p1.hashCode());
System.out.println("p2对象---------->"+p2.hashCode());
System.out.println("p1的地址---------->"+p1.address.hashCode());
System.out.println("p2的地址---------->"+p2.address.hashCode());
}
}
测试输出控制台结果:
可以看出p1作为模板克隆出了p2,而且p1和p2中的引用属性address也真正克隆出了一份
深拷贝(序列化)推荐使用
//person类实现序列化接口
//创建一个序列化克隆的方法getclone实现深克隆
public class Person implements Serializable {
String name;
int age;
Address address;
public Person(String name, int age, Address address) {
this.name = name;
this.age = age;
this.address = address;
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
", address=" + address +
'}';
}
public Person getClone(){
ByteArrayOutputStream bos = null;
ObjectOutputStream oos = null;
ByteArrayInputStream bis = null;
ObjectInputStream ois = null;
try {
//序列化
bos = new ByteArrayOutputStream(); //创建一个byte数组缓冲区
oos = new ObjectOutputStream(bos); //把byte缓冲区和对象对接上
oos.writeObject(this); //把对象写进byte缓冲区
//反序列化
bis = new ByteArrayInputStream(bos.toByteArray()); //把缓冲区的对象放进新的byte数组中
ois = new ObjectInputStream(bis); //把对象数组进过对象输入流处理成对象
//返回生成的新对象
Person person = (Person) ois.readObject();
return person;
} catch (Exception e) {
e.printStackTrace();
return null;
}finally {
try {
ois.close();
oos.close();
}catch (Exception e){
e.printStackTrace();
}
}
}
}
//Address类实现序列化接口
public class Address implements Serializable {
String city;
String street;
int emailNum;
public Address(String city, String street, int emailNum) {
this.city = city;
this.street = street;
this.emailNum = emailNum;
}
}
//测试
public class test {
public static void main(String[] args) throws Exception {
Address addr = new Address("广州市", "青云街", 40033);
Person p1 = new Person("小明", 18, addr);
Person p2 = p1.getClone();
System.out.println("p1---------------->"+p1);
System.out.println("p2---------------->"+p2);
System.out.println("p1哈希值---------------->"+p1.hashCode());
System.out.println("p2哈希值---------------->"+p2.hashCode());
System.out.println("p1的地址---------------->"+p1.address.hashCode());
System.out.println("p2的地址---------------->"+p2.address.hashCode());
}
}
深拷贝基本介绍
- 复制对象的所有基本数据类型的成员变量值
- 为所有引用数据类型的成员变量申请存储空间,并复制每个引用数据类型成员变量所引用的对象,直到该对象可达的所有对象。也就是说,对象进行深拷贝要对整个对象进行拷贝
- 深拷贝实现方式1:重写clone方法来实现深拷贝:
缺点: 当对象中有很多成员对象,且成员对象中又有成员对象时,使用clone的方式,需要在每个引用到的成员对象所在的类中重写clone方法,非常的麻烦。
- 深拷贝实现方式2:通过对象序列化实现深拷贝(推荐):
优点:序列化的方式,值需要在引用到的对象所在类实现Serializable接口,在写一个共用的复制对象的处理方法就行了,更加方便灵活。这是推荐的方式。
原型模式的注意事项和细节
- 创建新的对象比较复杂时,可以利用原型模式简化对象的创建过程,同时也能够提高效率
- 不用重新初始化对象,而是动态地获得对象运行时的状态
- 如果原始对象发生变化(增加或者减少属性),其它克隆对象的也会发生相应的变化,无需修改代码
- 在实现深克隆的时候可能需要比较复杂的代码
- 缺点:需要为每一个类配备一个克隆方法,这对全新的类来说不是很难,但对已有的类进行改造时,需要修改其源代码,违背了ocp原则,这点请同学们注意.