说到克隆,本质都是使用一个已经实例化完成的对象的副本。 对于基本类型比较简单。比方说我们想复制一个变量,
int index = 12345;
很简单,
int indexCopy = index;
如果我们想复制一个对象呢,情况就变的有点复杂
Student student1 = new Student();
student1.setName("Michael");
Student student2 = student1;
啧啧,作为新手,通常会会觉得这样的就完成了复制。 但实际情况是,当我们对 student1的 name进行更改时,student2的值就会被一起改变。
上面这个其实只是引用复制,大家都指向堆里的同一个引用,自然当数据发生变化时也会一起改变。 那么如何正确复制对象呢,这里就得说一下Java的Clone。
Java的克隆允许复制一个一模一样内容的对象出来,当改变A内容时,被克隆的B的内容不会一起改变。 为什么要克隆呢?其实很简单,当我们复制了一个对象的引用,新对象的数据在发生变更时会同时修改原对象的数据。而这并不是我们想要的,我们只想修改新对象的数据。克隆可以解决这种场景。 这里面有两种克隆,浅克隆和深克隆。
不管是浅克隆还是深克隆,都要先实现 Clonable接口,然后复写 clone()方法并改为 public。 举例有个 Student类
public class Student implements Clonebale{
String name;
@Override
public Object clone() {
Student stu = null;
try{
stu = (Student)super.clone();
}catch(CloneNotSupportedException e) {
e.printStackTrace();
}
return stu;
}
}
然后我们从student1复制一个对象student2出来,
student1.setName("Michael");
Student student2 = student1.clone();
student2.setName("Rachel");
这时候我们再打印两个对象的值,就会发现这是两个完全不同的对象,分别叫 Michael和 Rachel了。
当情况变的更复杂一点,比如在Student里引入一个对象 Subject
class Subject {
String title;
}
class Student implement Cloneable{
String name;
Subject subject;
}
这时候我们再用student1克隆出student2的结果就会变成,
name字段是一样的,然而subject字段却保持和student1
一样。
如果我们想把克隆对象里的非基本类型也一并克隆的话,那么需要把引用的类型也同样实现克隆接口。 也就是说,subject类也需要实现 Cloneable
class Subject implements Clonable{
String title;
@Override
public Object clone() {
....
}
}
这样便成为深克隆了。
如果我们想克隆的对象有多个层次的类型引用,这时候把每一个类型都实现 Clonable接口是不现实的。 那么可以用序列化和发序列化的方法来实现克隆。