1: Java中浅拷贝和深拷贝的定义:
浅拷贝:就是指两个对象共同拥有同一个值,一个对象改变了该值,也会影响到另一个对象。
深拷贝:就是两个对象的值相等,但是互相独立。
(深拷贝才是真正的拷贝,浅拷贝只是将引用指向了同一份对象)
2:Java中几种常见的拷贝操作:
(1)“=”操作:也就是赋值操作;
(2)拷贝构造函数:拷贝构造函数就是构造函数的参数的类型是该构造函数所在的类,即参数就是该类的一个对象。
<span style="font-size:18px;"><span style="font-size:18px;"><span style="font-size:18px;"> public Clock(Clock clock){
this.hour=clock.hour;
this.min=clock.min;
this.second=clock.second;
}</span></span></span>
(3) 实现 Cloneable接口,只是一个标记。
3:几种操作比较
= | 拷贝构造函数 | clone方法 | |
预定义非集合类型 (int ,Integer,String) | 1.深拷贝 | 2.如果支持拷贝构造函数的类型,则是深拷贝 | 3.不支持 |
自定义类型 (自定义的一个类) | 4.浅拷贝(一个改变,另外一个跟着改变) | 5.取决于实现() 深拷贝 | 6. 深拷贝 |
预定义集合类型 | 7.浅拷贝 | 8.会逐个调用每个元素的operator=方法 如果add元素是预定义数据类型,则为深拷贝 ; 如果add元素是自定义数据类型,则为浅 拷贝 | 9.会逐个调用每个元素的operator=方法 如果add元素是预定义数据类型,则为深拷贝 ; 如果add元素是自定义数据类型,则为浅 拷贝 |
4:如果一个类实现了clone()方法,并且该类中的成员变量既有基本数据类型(String,int);又有自定义类型的时候,则基本数据类型clone()的是深拷贝,自定义数据类型的是浅拷贝。
<span style="font-size:18px;"><span style="font-size:18px;">package shiyeqiang.resume;
public class ResumeDemo {
public static void main(String[] args) throws Exception {
Resume resumeA = new Resume("name A");
resumeA.setAge(10);
resumeA.setWorkExperience(1, "百度");
Resume resumeB = (Resume) resumeA.clone();
resumeB.setName("name B");
resumeB.setAge(20);
resumeB.setWorkExperience(2, "腾讯");
System.out.println(resumeA);
System.out.println(resumeB);
}
}
// 工作经历类
class WorkExperience {
private int workLong = 0;
private String company = null;
public WorkExperience() {
}
public int getWorkLong() {
return workLong;
}
public void setWorkLong(int workLong) {
this.workLong = workLong;
}
public String getCompany() {
return company;
}
public void setCompany(String company) {
this.company = company;
}
}
// 简历类中包括工作经历类以及其他基本数据类型成员
class Resume implements Cloneable {
private String name = null;
private int age = 0;
private WorkExperience workExperience = new WorkExperience();
public void setWorkExperience(int workLong, String company) {
this.workExperience.setWorkLong(workLong); // 设置工作时间
this.workExperience.setCompany(company); // 设置公司
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public Resume(String name) {
this.name = name;
}
public String toString() {
return "姓名=" + this.name + ", 年龄=" + this.age + ", 工作经历:工作年限="
+ this.workExperience.getWorkLong() + ", 公司名称="
+ this.workExperience.getCompany();
}
public Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
</span></span>
结果显示:
姓名=name A, 年龄=10,工作经历:工作年限=2, 公司名称=腾讯(浅拷贝)
姓名=name B, 年龄=20, 工作经历:工作年限=2, 公司名称=腾讯(浅拷贝)
5:那么一个类要想实现深拷贝应该怎么办呢?
让简历类Resume中引用成员 工作经历WorkExperience类也实现Cloneable接口,并写重写clone()方法;
修改Resume类中的clone方法,在其中必须调用WorkExperience的clone()方法;
其实说白了,深拷贝,深拷贝啊,也就是一层一层的拷贝,千万别直接用super.clone()方法偷懒哦!
<span style="font-size:18px;"><span style="font-size:18px;">package shiyeqiang.resume;
public class ResumeDemo {
public static void main(String[] args) throws Exception {
Resume resumeA = new Resume("name A");
resumeA.setAge(10);
resumeA.setWorkExperience(1, "百度");
Resume resumeB = (Resume) resumeA.clone();
resumeB.setName("name B");
resumeB.setAge(20);
resumeB.setWorkExperience(2, "腾讯");
System.out.println(resumeA);
System.out.println(resumeB);
}
}
// 工作经历类
class WorkExperience implements Cloneable{
private int workLong = 0;
private String company = null;
public WorkExperience() {
}
public int getWorkLong() {
return workLong;
}
public void setWorkLong(int workLong) {
this.workLong = workLong;
}
public String getCompany() {
return company;
}
public void setCompany(String company) {
this.company = company;
}
public Object clone() throws CloneNotSupportedException{
return super.clone();
}
}
// 简历类中包括工作经历类以及其他基本数据类型成员
class Resume implements Cloneable {
private String name = null;
private int age = 0;
private WorkExperience workExperience = null;
public void setWorkExperience(int workLong, String company) {
this.workExperience.setWorkLong(workLong); // 设置工作时间
this.workExperience.setCompany(company); // 设置公司
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public Resume(String name) {
this.name = name;
this.workExperience=new WorkExperience();
}
<span style="color:#ff0000;">private Resume(WorkExperience workExperience) throws CloneNotSupportedException{
this.workExperience=(WorkExperience) workExperience.clone();
}</span>
public String toString() {
return "姓名=" + this.name + ", 年龄=" + this.age + ", 工作经历:工作年限="
+ this.workExperience.getWorkLong() + ", 公司名称="
+ this.workExperience.getCompany();
}
<span style="color:#ff0000;">public Object clone() throws CloneNotSupportedException {
Resume obj=new Resume(this.workExperience); //调用私有构造函数中的</span><span style="color: rgb(255, 0, 0); background-color: inherit; font-family: Arial; ">workExperience</span><span style="color:#ff0000;">克隆方法
obj.age=this.age;
obj.name=name;
return obj;
}</span>
}
</span></span>
姓名=name A, 年龄=10, 工作经历:工作年限=1, 公司名称=百度
姓名=name B, 年龄=20, 工作经历:工作年限=2, 公司名称=腾讯