java object 之clone方法解析

一,首先来看一下源码

 1 protected native Object clone() throws CloneNotSupportedException; 

1、方法由native关键字修饰

  java中的native关键字表示这个方法是个本地方法,【java native说明】。而且native修饰的方法执行效率比非native修饰的高。

2、方法由protected修饰

  一个类在覆盖clone()方法时候,需要修改成public访问修饰符,这样才能保证其他所有的类都能够访问这个类的这个方法。

3、方法抛出CloneNotSupportedException异常

  一个类想要覆盖clone()方法,必须本身实现java.lang.Cloneable接口,否则会抛出CloneNotSupportedException异常。

二、clone()的作用

  注:我们这里的对象特指复杂类型的。

1、简单的=操作

  我们知道,java中的复杂类型的对象都是引用类型,他们往往存的都是对象的内存地址。因此我们不能仅仅通过 操作符这样简单的赋值操作。我们将一个对象a 赋值给另一个对象b ,我们仅仅是将对象a 的内存地址赋值给b ,使得他们两个对象都是指向的同一个内存地址。这样的后果是,对其中一个对象的修改之后都会影响到另一个对象。如下图表示:

1 Person p1 = new Person();
2 Person p2 = p1;

 

2、clone()

  使用clone()方法,可以快速的创建一个对象的副本,并且两个对象指向不同的内存地址。如下图表示:

1 Person p1 = new Person();
2 Person p2 = p1.clone();

 

三、shallow clone和deep clone

1、shallow clone(浅拷贝)

  shallow clone是指只clone对象本身,不clone对象里的字段。只调用super.clone(),只是shallow clone。虽然拷贝之后的对象是指向了不同的内存地址,但是对象里面的字段还是和之前的对象指向同一个内存地址。

 1 public class ShallowClone implements Cloneable {
 2 
 3     public String name;
 4     public int age;
 5     public Person person;
 6 
 7     public ShallowClone() {
 8     }
 9 
10     public ShallowClone(String name, int age, Person person) {
11         this.name = name;
12         this.age = age;
13         this.person = person;
14     }
15 
16     @Override
17     public ShallowClone clone() {
18         ShallowClone c = null;
19         try {
20             c = (ShallowClone) super.clone();
21             return c;
22         } catch (CloneNotSupportedException e) {
23             e.printStackTrace();
24         }
25         return c;
26     }
27 
28     public static void main(String[] args) {
29         Person p = new Person();
30         p.name = "p";
31         p.age = 10;
32 
33         ShallowClone c1 = new ShallowClone("Jim", 18, p);
34         System.out.printf("before clone: c1 = %s, c1.person =  %s\n", c1, c1.person);
35         ShallowClone c2 = c1.clone();
36         System.out.printf("after clone: c2 = %s, c2.person =  %s\n", c2, c2.person);
37     }
38 }

 

运行main()输出:

before clone: c1 = cre.sample.test.object.ShallowClone@558385e3, c1.person =  cre.sample.test.Person@2dcb25f1
after clone: c2 = cre.sample.test.object.ShallowClone@742808b3, c2.person =  cre.sample.test.Person@2dcb25f1

 

说明浅拷贝,ShallowClone对象内存地址改变了,但是对象里的Person字段内存地址没有改变;

2、deep clone(深拷贝)

  deep clone则是指在clone对象本身的同时,也clone对象里面的字段。

 1 /**
 2  * deep clone代码示例
 3  * Created by CreGu on 2016/6/9.
 4  */
 5 public class DeepClone implements Cloneable {
 6     public String name;
 7     public int age;
 8     public Person person;
 9 
10     public DeepClone() {
11     }
12 
13     public DeepClone(String name, int age, Person person) {
14         this.name = name;
15         this.age = age;
16         this.person = person;
17     }
18 
19     @Override
20     public DeepClone clone() {
21         DeepClone c = null;
22         try {
23             c = (DeepClone) super.clone();
24             c.person = person.clone();
25             return c;
26         } catch (CloneNotSupportedException e) {
27             e.printStackTrace();
28         }
29         return c;
30     }
31 
32     public static void main(String[] args) {
33         Person p = new Person();
34         p.name = "p";
35         p.age = 10;
36 
37         DeepClone c1 = new DeepClone("Jim", 18, p);
38         System.out.printf("before clone: c1 = %s, c1.person =  %s\n", c1, c1.person);
39         DeepClone c2 = c1.clone();
40         System.out.printf("after clone: c2 = %s, c2.person =  %s\n", c2, c2.person);
41     }
42 }

 

运行main()输出:

before clone: c1 = cre.sample.test.object.DeepClone@558385e3, c1.person = cre.sample.test.Person@2dcb25f1
after clone: c2 = cre.sample.test.object.DeepClone@742808b3, c2.person = cre.sample.test.Person@70535b58

 说明深拷贝,DeepClone对象内存地址改变了,但是对象里的Person字段内存地址也改变了;

posted @ 2016-06-08 12:36  CreGu  阅读(420)  评论(2编辑  收藏  举报