我们知道在Java中存在这个接口Cloneable,实现该接口的类都会具备被拷贝的能力,同时拷贝是在内存中进行,在性能方面比我们直接通过new生成对象来的快,特别是在大对象的生成上,使得性能的提升非常明显。然而我们知道拷贝分为深拷贝和浅拷贝之分,但是浅拷贝存在对象属性拷贝不彻底问题。
一、clone方法浅拷贝问题:
Java中对象的克隆,为了获取对象的一份拷贝,我们可以利用Object类的clone()方法。Object类里的clone方法是浅拷贝。
必须要遵循下面三点:
1.在派生类中覆盖基类的clone()方法,并声明为public【Object类中的clone()方法为protected的】。
2.在派生类的clone()方法中,调用super.clone()。
3.在派生类中实现Cloneable接口。
用Object类的clone()方法,但是该方法存在一个缺陷,他并不会将对象的所有属性全部拷贝过来,而是有选择性的拷贝,基本规则如下:
(1)基本类型:
如果变量是基本类型,则拷贝其值,比如Int、float等。
(2)对象:
如果变量是一个实例对象,则拷贝其地址引用,也就是说此时新对象与原来对象是公用该实例变量。
(3)String字符串:
如果变量为String字符串,则拷贝其引用地址,但是在修改的时候,它会从字符串池中重新生成一个新的字符串,原有的字符串对象保持不变。
二、利用序列化实现对象的拷贝:
利用序列化来做深复制,把对象写到流里的过程是序列化(Serilization)过程,而把对象从流中读出来的过程则叫做反序列化(Deserialization)过程。应当指出的是,写在流里的是对象的一个拷贝,而原对象仍然存在于JVM里面。利用这个特性,可以做深拷贝 。并且该新对象与母对象之间并不存在引用共享的问题,真正实现对象的深拷贝。
所以使用该工具类(CloneUtils)的对象只要实现Serializable接口就可实现对象的克隆,无须继承Cloneable接口实现clone()方法。
针对深浅拷贝的理解:
- 浅拷贝:对基本数据类型进行值传递,对引用数据类型复制一个引用指向原始引用的对象,就是复制的引用和原始引用指向同一个对象。
- 深拷贝:对基本数据类型进行值传递,对引用数据类型,创建一个新的对象,并复制其内容,两个引用对象指向两个对象,但对象内容相同。