浅拷贝小结:
Object对象有个clone()方法,实现了对象中各个属性的复制,但它的可见范围是protected的,所以实体类使用克隆的前提是:
① 实现Cloneable接口,这是一个标记接口,自身没有方法。
② 覆盖clone()方法,可见性提升为public。
如果一个被复制的属性都是基本类型,那么只需要实现当前类的cloneable机制就可以了,此为浅拷贝。
如果被复制对象的属性包含其他实体类对象引用,那么这些实体类对象都需要实现cloneable接口并覆盖clone()方法。
深拷贝小结
① 如果有一个非原生成员,如自定义对象的成员,那么就需要:该成员实现Cloneable接口并覆盖clone()方法,不要忘记提升为public可见。同时,修改被复制类的clone()方法,增加成员的克隆逻辑。
② 如果被复制对象不是直接继承Object,中间还有其它继承层次,每一层super类都需要实现Cloneable接口并覆盖clone()方法。
与对象成员不同,继承关系中的clone不需要被复制类的clone()做多余的工作。
一句话来说,如果实现完整的深拷贝,需要被复制对象的继承链、引用链上的每一个对象都实现克隆机制。
浅拷贝: 被复制对象的所有值属性都含有与原来对象的相同,而所有的对象引用属性仍然指向原来的对象。
package com.basis.entity;
public class Person implements Cloneable {
private String adress;
private Info info;
public String getAdress() {
return adress;
}
public void setAdress(String adress) {
this.adress = adress;
}
public Info getInfo() {
return info;
}
public void setInfo(Info info) {
this.info = info;
}
@Override
public String toString() {
return "Person{" +
"adress='" + adress + '\'' +
", info=" + info +
'}';
}
public Object clone() throws CloneNotSupportedException{
return super.clone();
}
}
package com.basis.entity;
public class Info {
private String IDCard;
private String name;
public String getIDCard() {
return IDCard;
}
public void setIDCard(String IDCard) {
this.IDCard = IDCard;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "Info{" +
"IDCard='" + IDCard + '\'' +
", name='" + name + '\'' +
'}';
}
}
package com.company;
import com.basis.entity.Info;
import com.basis.entity.Person;
public class Main {
public static void main(String[] args) {
Info info = new Info();
info.setName("张三");
info.setIDCard("111111");
Person person1 = new Person();
person1.setAdress("北京");
person1.setInfo(info);
System.out.println("clone()前person1:"+person1.toString());
Person person2 = null;
try {
person2 = (Person)person1.clone();
person2.setAdress("河南");
Info info1 = person2.getInfo();
info1.setName("李四");
info1.setIDCard("222222");
person2.setInfo(info1);
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
System.out.println("clone()后person1:"+person1.toString());
System.out.println("clone()后person2:"+person2.toString());
}
}
结果
clone()前person1:Person{adress='北京', info=Info{IDCard='111111', name='张三'}}
clone()后person1:Person{adress='北京', info=Info{IDCard='222222', name='李四'}}
clone()后person2:Person{adress='河南', info=Info{IDCard='222222', name='李四'}}
现象:
修改clone()后的那个对象person2的adress变量时,person1的adress变量没有发生修改。
但是修改clone()后的那个对象person2的Info变量时,而person1的Info变量也相应修改。
说明 :
当修改clone()后的对象时,修改的字段为引用类型,而不是基础数据类型时,会对clone()前的对象有所影响,
而这种实现的方式为浅拷贝(shadow copy)。
为了解决修改clone()得到的新对象不会影响被复制对象的字段的问题,我们需要实现深拷贝(deep copy)。
**深拷贝:**在浅拷贝的基础上,所有引用其他对象的变量也进行了clone,并指向被复制过的新对象
package com.basis.entity;
public class Person implements Cloneable {
private String adress;
private Info info;
public String getAdress() {
return adress;
}
public void setAdress(String adress) {
this.adress = adress;
}
public Info getInfo() {
return info;
}
public void setInfo(Info info) {
this.info = info;
}
@Override
public String toString() {
return "Person{" +
"adress='" + adress + '\'' +
", info=" + info +
'}';
}
public Object clone() throws CloneNotSupportedException{
Person person = (Person) super.clone();
person.info = (Info) info.clone();
return person;
}
}
package com.basis.entity;
public class Info implements Cloneable {
private String IDCard;
private String name;
public String getIDCard() {
return IDCard;
}
public void setIDCard(String IDCard) {
this.IDCard = IDCard;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "Info{" +
"IDCard='" + IDCard + '\'' +
", name='" + name + '\'' +
'}';
}
public Object clone() throws CloneNotSupportedException{
return super.clone();
}
}
package com.company;
import com.basis.entity.Info;
import com.basis.entity.Person;
public class Main {
public static void main(String[] args) {
Info info = new Info();
info.setName("张三");
info.setIDCard("111111");
Person person1 = new Person();
person1.setAdress("北京");
person1.setInfo(info);
System.out.println("clone()前person1:"+person1.toString());
Person person2 = null;
try {
person2 = (Person)person1.clone();
person2.setAdress("河南");
Info info1 = person2.getInfo();
info1.setName("李四");
info1.setIDCard("222222");
person2.setInfo(info1);
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
System.out.println("clone()后person1:"+person1.toString());
System.out.println("clone()后person2:"+person2.toString());
}
}
结果
clone()前person1:Person{adress='北京', info=Info{IDCard='111111', name='张三'}}
clone()后person1:Person{adress='北京', info=Info{IDCard='111111', name='张三'}}
clone()后person2:Person{adress='河南', info=Info{IDCard='222222', name='李四'}}