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
2
|
Person p1 = new Person();
Person p2 = p1;
|
2、clone()
使用clone()方法,可以快速的创建一个对象的副本,并且两个对象指向不同的内存地址。如下图表示:
1
2
|
Person p1 = new Person();
Person p2 = p1.clone();
|
三、shallow clone和deep clone1、shallow clone(浅拷贝)
shallow clone是指只clone对象本身,不clone对象里的字段。只调用super.clone(),只是shallow clone。虽然拷贝之后的对象是指向了不同的内存地址,但是对象里面的字段还是和之前的对象指向同一个内存地址。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
|
public class ShallowClone implements Cloneable {
public String name;
public int age;
public Person person;
public ShallowClone() {
}
public ShallowClone(String name, int age, Person person) {
this .name = name;
this .age = age;
this .person = person;
}
@Override
public ShallowClone clone() {
ShallowClone c = null ;
try {
c = (ShallowClone) super .clone();
return c;
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return c;
}
public static void main(String[] args) {
Person p = new Person();
p.name = "p" ;
p.age = 10 ;
ShallowClone c1 = new ShallowClone( "Jim" , 18 , p);
System.out.printf( "before clone: c1 = %s, c1.person = %s\n" , c1, c1.person);
ShallowClone c2 = c1.clone();
System.out.printf( "after clone: c2 = %s, c2.person = %s\n" , c2, c2.person);
}
}
|
运行main()输出:
1
2
|
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
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
|
/**
* deep clone代码示例
* Created by CreGu on 2016/6/9.
*/
public class DeepClone implements Cloneable {
public String name;
public int age;
public Person person;
public DeepClone() {
}
public DeepClone(String name, int age, Person person) {
this .name = name;
this .age = age;
this .person = person;
}
@Override
public DeepClone clone() {
DeepClone c = null ;
try {
c = (DeepClone) super .clone();
c.person = person.clone();
return c;
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return c;
}
public static void main(String[] args) {
Person p = new Person();
p.name = "p" ;
p.age = 10 ;
DeepClone c1 = new DeepClone( "Jim" , 18 , p);
System.out.printf( "before clone: c1 = %s, c1.person = %s\n" , c1, c1.person);
DeepClone c2 = c1.clone();
System.out.printf( "after clone: c2 = %s, c2.person = %s\n" , c2, c2.person);
}
}
|
运行main()输出:
1
2
|
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字段内存地址也改变了。
以上这篇java object 之clone方法全面解析就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持服务器之家。