Java的克隆
   来源:enjoyeduKyle     2019年04月13日 10:58

说到克隆,本质都是使用一个已经实例化完成的对象的副本。

对于基本类型比较简单。比方说我们想复制一个变量,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姐妹吗?

对象 就会 类型