介绍
- 原型模式(Prototype模式)是指:用原型实例指定创建对象的种类,并且通过拷贝这些原型,创建新的对象
- 原型模式是一种创建型设计模式,允许一个对象再创建另外一个可定制的对象,无需知道如何创建的细节
- 工作原理是:通过将一个原型对象传给那个要发动创建的对象,这个要发动创建的对象通过请求原型对象拷贝它们自己来实施创建,即 对象.clone()
个人总结
如果需要多个相同的定制化对象时(除了对象内存地址不一样),并不需要多次以相同方法创建,这个操作复杂,只需要调用对象.clone()克隆一个这样的对象即可
浅拷贝
//一个person类实现Cloneable接口的clone方法//person具有两个基本数据类型的属性name age和一个引用类型的属性addresspublic 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;}@Overridepublic String toString() {return "Person{" +"name='" + name + '\'' +", age=" + age +", address=" + address +'}';}@Overrideprotected 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;}@Overridepublic String toString() {return "Person{" +"name='" + name + '\'' +", age=" + age +", address=" + address +'}';}@Overrideprotected 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;}@Overrideprotected 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;}@Overridepublic 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原则,这点请同学们注意.
