原型模式——浅复制与深复制

原型模式涉及一个浅复制和深复制的概念。原型模式可以简单理解为“复制”,但这个复制不是代码的复制。对同一个类,我们可以实例化new三次来“复制”,但如果在初始化的时候构造函数的执行很长,多次实例化就显得效率很低效了。那我们能否只实例化一次,然后“复制”呢?

Test test1 = new Test();
Test test2 = test1;
Test test3 = test1;

这样写吗?注意这是引用的复制,这实际上还是只有test1一个实例,test2、test3只是复制了其引用而已,如果修改了一个对象则会影响到其他的对象。这样是不符合我们要求的。这就会引出我们Java的clone方法浅复制和深复制了。

我们先来看浅复制是什么。对于要实现克隆(我们后面将浅复制和深复制统称为克隆),必须实现Cloneable接口,尽管clone方法在Object类里,但我们还是得实现Cloneable接口不然会抛出CloneNotSupportedException错误。

定义一个Resume类实现Cloneable接口,并实现Object类中的clone方法。

 1 package day_12_prototype;
 2
 3 /**
 4  * @author turbo
 5  *
 6  * 2016年9月17日
 7  */
 8 public class Resume implements Cloneable {
 9     private String name;
10     private String sex;
11
12     public String getName() {
13         return name;
14     }
15
16     public void setName(String name) {
17         this.name = name;
18     }
19
20     public String getSex() {
21         return sex;
22     }
23
24     public void setSex(String sex) {
25         this.sex = sex;
26     }
27
28     @Override
29     protected Object clone() throws CloneNotSupportedException {
30         Resume resume = (Resume)super.clone();
31         return resume;
32     }
33
34 }

客户端测试代码:

 1 package day_12_prototype;
 2
 3 /**
 4  * @author turbo
 5  *
 6  * 2016年9月17日
 7  */
 8 public class Main {
 9
10     /**
11      * @param args
12      * @throws CloneNotSupportedException
13      */
14     public static void main(String[] args) throws CloneNotSupportedException {
15         Resume resume1 = new Resume();
16         Resume resume2 = (Resume)resume1.clone();
17         System.out.println(resume1.hashCode() + " " + resume2.hashCode());
18     }
19
20 }

我们可以看看输出结果,resume2是否只是复制了resume1的引用。

看来并不是。并且我们也成功地克隆了一个对象,并不是简单地复制了引用。我们的Resume类里只有两个String基本类型,我们来看看如果有其他引用会是什么样子的。

 1 package day_12_prototype;
 2
 3 /**
 4  * @author turbo
 5  *
 6  * 2016年9月17日
 7  */
 8 public class Resume implements Cloneable {
 9     private String name;
10     private String sex;
11     private Test test = new Test();
12
13     public String getName() {
14         return name;
15     }
16
17     public void setName(String name) {
18         this.name = name;
19     }
20
21     public String getSex() {
22         return sex;
23     }
24
25     public void setSex(String sex) {
26         this.sex = sex;
27     }
28
29     public Test getTest() {
30         return test;
31     }
32
33     public void setTest(Test test) {
34         this.test = test;
35     }
36
37     @Override
38     protected Object clone() throws CloneNotSupportedException {
39         Resume resume = (Resume)super.clone();
40         return resume;
41     }
42
43 }

此时我们增加了一个Test类的引用,其他代码不变,Test类里什么都没有。看看客户端代码:

 1 package day_12_prototype;
 2
 3 /**
 4  * @author turbo
 5  *
 6  * 2016年9月17日
 7  */
 8 public class Main {
 9
10     /**
11      * @param args
12      * @throws CloneNotSupportedException
13      */
14     public static void main(String[] args) throws CloneNotSupportedException {
15         Resume resume1 = new Resume();
16         Resume resume2 = (Resume)resume1.clone();
17         System.out.println(resume1.hashCode() + " " + resume2.hashCode());
18         System.out.println(resume1.getTest().hashCode() + " " + resume2.getTest().hashCode());
19     }
20
21 }

这个时候对第18行的输出结果会是什么样的呢?

我们看到虽然我们对resume1进行了克隆,resume2确实也是新的引用,但由于Resume类中有了对另外一个类的引用,所以resume1和resume2对Test对象的引用还是同一个,这就是浅复制。那么如何做到连同Test对象一起克隆,而不是只复制一个引用呢?这就是深复制的概念。

我们首先对在Resume中被引用的类Test做一点改变,让它实现Cloneable接口,实现clone方法:

 1 package day_12_prototype;
 2
 3 /**
 4  * @author turbo
 5  *
 6  * 2016年9月17日
 7  */
 8 public class Test implements Cloneable{
 9     private String test;
10
11     public String getTest() {
12         return test;
13     }
14
15     public void setTest(String test) {
16         this.test = test;
17     }
18
19     @Override
20     protected Object clone() throws CloneNotSupportedException {
21         return super.clone();
22     }
23
24 }

同样在Resume类中做一点微小的改变:

 1 package day_12_prototype;
 2
 3 /**
 4  * @author turbo
 5  *
 6  * 2016年9月17日
 7  */
 8 public class Resume implements Cloneable {
 9     private String name;
10     private String sex;
11     private Test test = new Test();
12
13     public String getName() {
14         return name;
15     }
16
17     public void setName(String name) {
18         this.name = name;
19     }
20
21     public String getSex() {
22         return sex;
23     }
24
25     public void setSex(String sex) {
26         this.sex = sex;
27     }
28
29     public Test getTest() {
30         return test;
31     }
32
33     public void setTest(Test test) {
34         this.test = test;
35     }
36
37     @Override
38     protected Object clone() throws CloneNotSupportedException {
39         Resume resume = (Resume)super.clone();
40         resume.test = (Test)test.clone();
41         return resume;
42     }
43
44 }

客户端测试代码不变:

 1 package day_12_prototype;
 2
 3 /**
 4  * @author turbo
 5  *
 6  * 2016年9月17日
 7  */
 8 public class Main {
 9
10     /**
11      * @param args
12      * @throws CloneNotSupportedException
13      */
14     public static void main(String[] args) throws CloneNotSupportedException {
15         Resume resume1 = new Resume();
16         Resume resume2 = (Resume)resume1.clone();
17         System.out.println(resume1.hashCode() + " " + resume2.hashCode());
18         System.out.println(resume1.getTest().hashCode() + " " + resume2.getTest().hashCode());
19     }
20
21 }

输出结果:

这样我们就实现了对象的深复制。

说完浅复制与深复制,其实我们也就讲完了原型模式:用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。为什么要通过克隆的方式来创建新的对象,也即是我们在上面提到的,每new一次都需要执行一次构造函数,如果构造函数的执行时间很长,那么多次执行这个初始化操作就实在是太低效了。

时间: 2024-10-17 03:25:51

原型模式——浅复制与深复制的相关文章

原型模式——浅复制和深复制

设计模式这本书通过对小菜求职复印简历,给我们引出了原型模式:用原型实例制定创建对象的种类,并且通过拷贝这些原型创建新的对象.其实说简单点就是我们创建好一个模板,然后进行复制,或者将模板修改后进行制-- 由于MemberwiseClone()方法对于字段为值类型的,则对该字段执行逐位复制:如果字段是引用类型,则复制引用但不复制引用的对象,因此原始对象及其复本引用同一对象.所以我们把复制分为浅复制和深复制! 浅复制:被复制对象的所有变量都含有与原来的对象相同的值,而其所有的对其他对象的引用都仍然指向

原型模式——浅复制VS深复制

在学习设计模式中的原型模式中遇到了浅复制和深复制这两个陌生的面孔.它们到底是什么意思?浅和深到底是什么意思?还是需要来仔细研究一下的 首先,了解一下原型模式:用原型实例制定创建对象的种类,并且通过拷贝这些原型创建新的对象.通俗的将其实就是复制. 若收到复制,并不陌生,但是怎么用,我们又该如何用才能达到更加灵活的复制呢? 浅复制: 1.首先要实现最简单的复制,我们用到的一个Clone的方法,具体用法如下: public override Prototype Clone() { return thi

也来谈一谈js的浅复制和深复制

1.浅复制VS深复制 本文中的复制也可以称为拷贝,在本文中认为复制和拷贝是相同的意思.另外,本文只讨论js中复杂数据类型的复制问题(Object,Array等),不讨论基本数据类型(null,undefined,string,number和boolean),这些类型的值本身就存储在栈内存中(string类型的实际值还是存储在堆内存中的,但是js把string当做基本类型来处理 ),不存在引用值的情况. 浅复制和深复制都可以实现在已有对象的基础上再生一份的作用,但是对象的实例是存储在堆内存中然后通

python基础之浅复制与深复制

关于列表.字典.元组的浅复制和深复制,数字和字符串没有浅复制和深复制一说.下面我直接用代码来体现: 1 import copy 2 3 names = ['Cahill','Teenglan','Eric','Peggie','Aalto','Baal','Sadie', 4 'Gage','Hagan','Jack','Kaley','Mabel','Lacy','Nadine','Pace','Amy'] 5 6 #浅复制 7 names2 = names.copy() 8 names2[1

JAVA浅复制与深复制

1.浅复制与深复制概念 ⑴浅复制(浅克隆)     多个变量指向一个对象    被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用仍然指向原来的对象.换言之,浅复制仅仅复制所考虑的对象,而不复制它所引用的对象. ⑵深复制(深克隆)     每个变量指向一个对象,同时对象内包含对象,能复制内部对象    被复制对象的所有变量都含有与原来的对象相同的值,除去那些引用其他对象的变量.那些引用其他对象的变量将指向被复制过的新对象,而不再是原有的那些被引用的对象.换言之,深复制把要复

转载:python中的copy模块(浅复制和深复制)

主要是介绍python中的copy模块. copy模块包括创建复合对象(包括列表.元组.字典和用户定义对象的实例)的深浅复制的函数. ########copy(x)########创建新的复合对象并通过引用复制x的成员来创建x的浅复制.更加深层次说,它复制了对象,但对于对象中的元素,依然使用引用.对于内置类型,此函数并不经常使用.而是使用诸如list(x), dict(x), set(x)等调用方式来创建x的浅复制,要知道像这样直接使用类型名显然比使用copy()快很多.但是它们达到的效果是一样

深度解析javascript中的浅复制和深复制

原文:深度解析javascript中的浅复制和深复制 在谈javascript的浅复制和深复制之前,我们有必要在来讨论下js的数据类型.我们都知道有Number,Boolean,String,Null,Undefined,Object五种类型.而Object又包含Function,Array和Object自身.前面的五种类型叫做基本类型,而Object是引用类型.可能有人就要问,为什么要分基本类型和引用类型呢?后面你就会明白的. 我们首先来看看浅复制和深复制的简洁定义: 深复制:直接将数据复制给

c++中浅复制与深复制

在C++中经常会遇到有关类对象的浅复制与深复制的问题,也是容易出错的地方. 查找了相关资料,有关浅复制与深复制的定义为:对类进行复制的时候按位复制,即把一个对象各数据成员的值原样复制到目标对象中.当类中涉及到指针类型数据成员的时候,往往就会产生指针悬挂问题. class A{ public: int* a; }; A a1; A b1=a1; b1=a1执行的是浅复制,此时a1.a和b1.a指向的是同一个内存地址,如果在析构函数里面有对内存的释放.就会出现内存访问异常.因为一块内存空间会被释放两

java基础-浅复制与深复制的理解

浅复制与深复制在很多编程语言中都有出现,那么什么是浅复制,什么是深复制呢? 要区分浅复制与深复制,首先我们要明确什么是复制,怎样才算是复制.复制的例子在生活中也随处可见,如复印一份文档,复制一段文字等.我们可以发现,复制操作后可以得到两份相同的东西,即复制由一变为二了.下面来看一个例子: public class User{ private int age; public int getAge(){ return age; } } User user1 = new User(); User us