在看solr源码的时候看到一段使用clone的代码,一时想不起来clone的相关内容。所以这里做一个记录总结。
首先,clone的作用是对复制一个对象的内容,复制的新对象相对于原对象独立。对新对象的修改理论上不会影响原对象。
然后是clone的使用方法,需要先实现Cloneable接口,相当于取得使用clone的权限。然后根据自己的需求重写clone方法就行了。
clone的使用方法比较简单,接下来写个例子具体了解一下:
public class Test {
public static class State implements Cloneable {
private Integer i;
State o;
public void setI(Integer i) {
this.i = i;
}
public Integer getI() {
return this.i;
}
public State() {
}
public State(Integer i) {
this.i = i;
}
// 重写方法
@Override
public State clone() {
State clone = new State();
try {
clone = (State) super.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return clone;
}
}
}
这里没有其它的需求,所以直接用super.clone()
复制对象返回就行了,捕捉的异常发生在没有实现接口的前提下使用clone方法。
然后我们试着操作一下:
public static void main(String[] args) {
State state = new State(1);
State o = new State(2);
state.o = o;
// 克隆对象
State state2 = state.clone();
System.out.println(state2.getI());
// 修改克隆对象的属性
state2.setI(2);
state2.o.setI(3);
// 输出原对象的属性
System.out.println(state.getI());
System.out.println(state.o.getI());
}
克隆一个对象后,对克隆对象的属性值修改,看一下原对象是否会变化:
上图是输出结果,第一行表示克隆对象的值与原对象一样;第二行第三行是在对克隆对象修改后的输出,克隆对象的属性i改为2,属性o的属性i改为了3,而原对象的属性i没变,但属性o的属性i变了。
这就是前文说到的,对克隆对象的修改理论上不影响原对象。
这里涉及到了clone的一个概念:浅复制/深复制
Java默认的clone属于浅复制,就前面的例子来说,类State的两个属性一个是基本类型,一个是对象。那么调用clone方法实际上复制的是基本类型的值和对象的索引值。所以原对象和克隆对象的属性o指向同一个对象,那么对克隆对象属性o的修改自然会影响原对象的属性o。
深复制的意思就是,针对类中对象类型的属性也使用clone方法复制一下,这样克隆对象的修改基本就不会影响到原对象了。前提是,对象类型的属性中没有对象类型的属性。
说得有点绕口,这里有篇说的很详细的文章