- clone的过程
new 一个对象的过程和 clone 一个对象的过程区别
new 操作符的本意是分配内存。程序执行到 new 操作符时, 首先去看 new 操作符后面的类型,因为知道了类型, 才能知道要分配多大的内存空间。分配完内存之后,再调用构造函数,填充对象的各个域,这一步叫做对象的初始化, 构造方法返回后,一个对象创建完毕,可以把他的引用(地址)发布到外部,在外部就可以使用这个引用操纵这个对 象。
clone 在第一步是和 new 相似的, 都是分配内存,调用 clone 方法时,分配的内存和原对象(即调用 clone 方 法的对象)相同,然后再使用原对象中对应的各个域,填充新对象的域, 填充完成之后,clone 方法返回,一个新的 相同的对象被创建,同样可以把这个新对象的引用发布到外部。
即相同点是:都会重新分配内存;不同点是clone的数据和原数据相同,而new的话数据是初始数据
- 引用复制
是一个对象,地址相同
A a=new A();
A b=a;
- 对象复制
不同对象,地址不同
前提:class A 需实现cloneable
且实现clone(){return super.clone()}
A a=new A();
A b=a.clone();
其中对于对象复制,分为浅复制和深复制
举个例子
class A implement Cloneable{
int a;
String str;
C obj;
clone(){
return super.clone()
}
}
其中 默认的clone行为:super.clone() 对于基本数据对象是对象复制,而对于string及对象都是引用复制;
因此:
如果调用:
A a=new A();
B b=a.clone();
因为a实现了克隆接口,故a,b 是对象复制,是不同的对象;
但由于a只是实现了默认的克隆实现,因此
a.str==b.str
a.obj==b.obj
即a只是浅拷贝;
那a怎么实现深度拷贝呢? 那就需要改变默认的拷贝行为
class A{
...
clone(){
A copyA=(A)super.clone();
copyA.str=new String(str);
copyA.obj=(C)obj.clone();
return copyA;
}
}
这样的话,即在class A这个层级实现了深拷贝
即:a.str != b.str
a.obj != b.obj
为什么说在 class A这个层级实现了深拷贝,而不是就是深拷贝了呢?
因为 A中的对象 C ,它的默认拷贝实现是深拷贝还是浅拷贝得看它的变量类型以及拷贝实现;