介绍

  1. 原型模式(Prototype模式)是指:用原型实例指定创建对象的种类,并且通过拷贝这些原型,创建新的对象
  2. 原型模式是一种创建型设计模式,允许一个对象再创建另外一个可定制的对象,无需知道如何创建的细节
  3. 工作原理是:通过将一个原型对象传给那个要发动创建的对象,这个要发动创建的对象通过请求原型对象拷贝它们自己来实施创建,即 对象.clone()

个人总结
如果需要多个相同的定制化对象时(除了对象内存地址不一样),并不需要多次以相同方法创建,这个操作复杂,只需要调用对象.clone()克隆一个这样的对象即可

浅拷贝

  1. //一个person类实现Cloneable接口的clone方法
  2. //person具有两个基本数据类型的属性name age和一个引用类型的属性address
  3. public class Person implements Cloneable{
  4. String name;
  5. int age;
  6. Address address;
  7. public Person(String name, int age, Address address) {
  8. this.name = name;
  9. this.age = age;
  10. this.address = address;
  11. }
  12. @Override
  13. public String toString() {
  14. return "Person{" +
  15. "name='" + name + '\'' +
  16. ", age=" + age +
  17. ", address=" + address +
  18. '}';
  19. }
  20. @Override
  21. protected Object clone() throws CloneNotSupportedException {
  22. return super.clone();
  23. }
  24. }
  25. //Address类
  26. public class Address {
  27. String city;
  28. String street;
  29. int emailNum;
  30. public Address(String city, String street, int emailNum) {
  31. this.city = city;
  32. this.street = street;
  33. this.emailNum = emailNum;
  34. }
  35. }
  36. //测试
  37. public class test {
  38. public static void main(String[] args) throws Exception {
  39. Address addr = new Address("广州市", "青云街", 40033);
  40. Person p1 = new Person("小明", 18, addr);
  41. Person p2 = (Person)p1.clone();
  42. System.out.println(p1);
  43. System.out.println(p2);
  44. System.out.println("p1对象---------->"+p1.hashCode());
  45. System.out.println("p2对象---------->"+p2.hashCode());
  46. System.out.println("p1的地址---------->"+p1.address.hashCode());
  47. System.out.println("p2的地址---------->"+p2.address.hashCode());
  48. }
  49. }

测试输出控制台结果:
可以看出虽然使用p1为模板克隆出了一个p2,但是p1和p2的引用属性address并没有真正的克隆,他们只是指向了一个相同的address对象image-20200925194552886.png

深拷贝(重写clone方法)

  1. //一个person类实现Cloneable接口的clone方法
  2. //person具有两个基本数据类型的属性name age和一个引用类型的属性address
  3. //实现深拷贝需要重写clone方法
  4. public class Person implements Cloneable{
  5. String name;
  6. int age;
  7. Address address;
  8. public Person(String name, int age, Address address) {
  9. this.name = name;
  10. this.age = age;
  11. this.address = address;
  12. }
  13. @Override
  14. public String toString() {
  15. return "Person{" +
  16. "name='" + name + '\'' +
  17. ", age=" + age +
  18. ", address=" + address +
  19. '}';
  20. }
  21. @Override
  22. protected Object clone() throws CloneNotSupportedException {
  23. Object obj = super.clone(); //首先把基本数据类型的属性克隆一次
  24. Person person = (Person)obj;
  25. person.address=(Address) address.clone(); //在把引用类型的基本数据类型复制一次
  26. return person;
  27. }
  28. }
  29. //Address类需要实现Cloneable的clone方法
  30. public class Address implements Cloneable{
  31. String city;
  32. String street;
  33. int emailNum;
  34. public Address(String city, String street, int emailNum) {
  35. this.city = city;
  36. this.street = street;
  37. this.emailNum = emailNum;
  38. }
  39. @Override
  40. protected Object clone() throws CloneNotSupportedException {
  41. return super.clone();
  42. }
  43. }
  44. //测试
  45. public class test {
  46. public static void main(String[] args) throws Exception {
  47. Address addr = new Address("广州市", "青云街", 40033);
  48. Person p1 = new Person("小明", 18, addr);
  49. Person p2 = (Person)p1.clone();
  50. System.out.println(p1);
  51. System.out.println(p2);
  52. System.out.println("p1对象---------->"+p1.hashCode());
  53. System.out.println("p2对象---------->"+p2.hashCode());
  54. System.out.println("p1的地址---------->"+p1.address.hashCode());
  55. System.out.println("p2的地址---------->"+p2.address.hashCode());
  56. }
  57. }

测试输出控制台结果:
可以看出p1作为模板克隆出了p2,而且p1和p2中的引用属性address也真正克隆出了一份image-20200925195918647.png

深拷贝(序列化)推荐使用

  1. //person类实现序列化接口
  2. //创建一个序列化克隆的方法getclone实现深克隆
  3. public class Person implements Serializable {
  4. String name;
  5. int age;
  6. Address address;
  7. public Person(String name, int age, Address address) {
  8. this.name = name;
  9. this.age = age;
  10. this.address = address;
  11. }
  12. @Override
  13. public String toString() {
  14. return "Person{" +
  15. "name='" + name + '\'' +
  16. ", age=" + age +
  17. ", address=" + address +
  18. '}';
  19. }
  20. public Person getClone(){
  21. ByteArrayOutputStream bos = null;
  22. ObjectOutputStream oos = null;
  23. ByteArrayInputStream bis = null;
  24. ObjectInputStream ois = null;
  25. try {
  26. //序列化
  27. bos = new ByteArrayOutputStream(); //创建一个byte数组缓冲区
  28. oos = new ObjectOutputStream(bos); //把byte缓冲区和对象对接上
  29. oos.writeObject(this); //把对象写进byte缓冲区
  30. //反序列化
  31. bis = new ByteArrayInputStream(bos.toByteArray()); //把缓冲区的对象放进新的byte数组中
  32. ois = new ObjectInputStream(bis); //把对象数组进过对象输入流处理成对象
  33. //返回生成的新对象
  34. Person person = (Person) ois.readObject();
  35. return person;
  36. } catch (Exception e) {
  37. e.printStackTrace();
  38. return null;
  39. }finally {
  40. try {
  41. ois.close();
  42. oos.close();
  43. }catch (Exception e){
  44. e.printStackTrace();
  45. }
  46. }
  47. }
  48. }
  49. //Address类实现序列化接口
  50. public class Address implements Serializable {
  51. String city;
  52. String street;
  53. int emailNum;
  54. public Address(String city, String street, int emailNum) {
  55. this.city = city;
  56. this.street = street;
  57. this.emailNum = emailNum;
  58. }
  59. }
  60. //测试
  61. public class test {
  62. public static void main(String[] args) throws Exception {
  63. Address addr = new Address("广州市", "青云街", 40033);
  64. Person p1 = new Person("小明", 18, addr);
  65. Person p2 = p1.getClone();
  66. System.out.println("p1---------------->"+p1);
  67. System.out.println("p2---------------->"+p2);
  68. System.out.println("p1哈希值---------------->"+p1.hashCode());
  69. System.out.println("p2哈希值---------------->"+p2.hashCode());
  70. System.out.println("p1的地址---------------->"+p1.address.hashCode());
  71. System.out.println("p2的地址---------------->"+p2.address.hashCode());
  72. }
  73. }

测试输出控制台结果:
p1通过序列化克隆的方法把p2克隆出来image-20200925205035594.png

深拷贝基本介绍

  1. 复制对象的所有基本数据类型的成员变量值
  2. 为所有引用数据类型的成员变量申请存储空间,并复制每个引用数据类型成员变量所引用的对象,直到该对象可达的所有对象。也就是说,对象进行深拷贝要对整个对象进行拷贝
  3. 深拷贝实现方式1:重写clone方法来实现深拷贝:

缺点: 当对象中有很多成员对象,且成员对象中又有成员对象时,使用clone的方式,需要在每个引用到的成员对象所在的类中重写clone方法,非常的麻烦。

  1. 深拷贝实现方式2:通过对象序列化实现深拷贝(推荐):

优点:序列化的方式,值需要在引用到的对象所在类实现Serializable接口,在写一个共用的复制对象的处理方法就行了,更加方便灵活。这是推荐的方式。

原型模式的注意事项和细节

  1. 创建新的对象比较复杂时,可以利用原型模式简化对象的创建过程,同时也能够提高效率
  2. 不用重新初始化对象,而是动态地获得对象运行时的状态
  3. 如果原始对象发生变化(增加或者减少属性),其它克隆对象的也会发生相应的变化,无需修改代码
  4. 在实现深克隆的时候可能需要比较复杂的代码
  5. 缺点:需要为每一个类配备一个克隆方法,这对全新的类来说不是很难,但对已有的类进行改造时,需要修改其源代码,违背了ocp原则,这点请同学们注意.