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