深拷贝和浅拷贝最根本的区别在于是否真正获取一个对象的复制实体,而不是引用。
假设B复制了A,修改A的时候,看B是否发生变化:
如果B跟着也变了,说明是浅拷贝,拿人手短!(修改堆内存中的同一个值)
如果B没有改变,说明是深拷贝,自食其力!(修改堆内存中的不同的值)
1、浅拷贝
在拷贝一个对象时,对对象的基本数据类型的成员变量进行拷贝,但对引用类型的成员变量只进行引用的传递,并没有创建一个新的对象,当对引用类型的内容修改会影响被拷贝的对象。
只是增加了一个指针指向已存在的内存地址,java中clone方法是一个浅拷贝,引用类型依然在传递引用。
如果克隆对象的子对象是不可变的,或者子对象没有更改器方法,那么就是安全的
2、深拷贝
除了对基本数据类型的成员变量进行拷贝,对引用类型的成员变量进行拷贝时,创建一个新的对象来保存引用类型的成员变量。
增加了一个指针并且申请了一个新的内存,使这个增加的指针指向这个新的内存。
重新定义clone方法,克隆出所有子对象
Cloneable接口出现和接口的使用没有任何关系,因为clone方法是Object类继承而来的
Cloneable接口是标记接口,不含任何方法,唯一的作用就是允许在类型查询中视同instanceof
实现深拷贝有两种方法:
(1)序列化该对象,然后反序列化回来,就能得到一个新的对象了。
序列化:将对象写入到IO流中; 反序列化:从IO流中恢复对象 序列化机制允许将实现序列化的java对象转化为字节序列,这些字节序列可以保存到磁盘或者网络传输上,以达到以后恢复成原来的对象,序列化机制使得对象可以脱离程序的运行而独立存在。
(2)继续利用clone()方法,对该对象的引用类型变量再实现一次clone()方法。
(1)序列化
public class Student3 implements Serializable,Cloneable{
2 private static final long serialVersionUID = 3462139480068147262L;
3 private Integer age;
4 private String name;
5
6 public Student3(Integer age, String name) {
7 this.age = age;
8 this.name = name;
9 }
10
11 public Integer getAge() {
12 return age;
13 }
14
15 public void setAge(Integer age) {
16 this.age = age;
17 }
18
19 public String getName() {
20 return name;
21 }
22
23 public void setName(String name) {
24 this.name = name;
25 }
26
27 @Override
28 protected Object clone() throws CloneNotSupportedException {
29 return super.clone();
30 }
31
32 public static void main(String[] args) throws CloneNotSupportedException {
33 File file = new File("D:/test.txt");
34 Student3 stu = new Student3(18, "xiaoxian");
35
36 System.out.println("clone方法是浅拷贝");
37 Student3 clone = (Student3)stu.clone();
38 System.out.println("clone == stu的结果:"+ (clone==stu));
39 System.out.println("clone.name == stu.name的结果:"+ (clone.name==stu.name));
40
41 System.out.println("将对象序列化是深拷贝");
42 //将对象序列化到IO流中
43 try {
44 ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream(file));
45 objectOutputStream.writeObject(stu);
46 objectOutputStream.close();
47 } catch (IOException e) {
48 e.printStackTrace();
49 }
50
51 //将对象从IO流中反序列化出来
52 try {
53 ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream(file));
54 Student3 student3 = (Student3) objectInputStream.readObject();
55 System.out.println("student3 == stu的结果:"+(stu == student3));
56 System.out.println("student3.name == stu.name的结果:"+(stu.name == student3.name));
57 } catch (Exception e) {
58 e.printStackTrace();
59 }
60 }
61
62 }
(2)重写clone方法
package com.company.DeepCopy;
public class DeepCopy {
public static void main(String[] args) {
Age a = new Age(20);
Student stu1 = new Student("宁采臣",a,174);
//通过调用重写后的clone方法进行浅拷贝
Student stu2 = (Student)stu1.clone();
System.out.println(stu1.toString());
System.out.println(stu2.toString());
System.out.println();
//尝试修改stu1中的各属性,观察stu2的属性有没有变化
stu1.setName("聂小倩");
//改变age这个引用类型的成员变量的值
a.setAge(18);
//stu1.setaAge(new Age(99)); 使用这种方式修改age属性值的话,stu2是不会跟着改变的。因为创建了一个新的Age类对象而不是改变原对象的实例值
stu1.setLength(157);
System.out.println(stu1.toString());
System.out.println(stu2.toString());
}
}
package com.company.DeepCopy;
public class Student implements Cloneable {
//学生类的成员变量(属性),其中一个属性为类的对象
private String name;
private Age aage;
private int length;
//构造方法,其中一个参数为另一个参数的对象
public Student(String name,Age a,int length){
this.name = name;
this.aage = a;
this.length = length;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Age getAge() {
return this.aage;
}
public void setAge(Age age) {
this.aage = age;
}
public int getLength() {
return this.length;
}
public void setLength(int length) {
this.length = length;
}
public String toString(){
return "姓名是:"+this.getName()+",年龄为:"+this.getAge().toString()+",长度是:"+this.getLength();
}
//重写Object类的clone方法
public Object clone(){
Object obj = null;
//调用Object类的clone方法---浅拷贝
try {
obj = super.clone();
}catch (CloneNotSupportedException e){
e.printStackTrace();
}
//调用Age类的clone方法进行深拷贝
//先将obj转化为学生类实例
Student stu = (Student)obj;
//学生类实例的Age对象属性,调用其clone方法进行拷贝
stu.aage=(Age)stu.getAge().clone();
return obj;
}
}
package com.company.DeepCopy;
public class Age implements Cloneable{
//年龄类的成员变量(属性)
private int age;
//构造方法
public Age(int age){
this.age = age;
}
public int getAge(){
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return this.age+"";
}
//重写Object的clone方法
public Object clone() {
Object obj = null;
try {
obj = super.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return obj;
}
}