what——克隆是什么?
克隆起源于生物学科,大致意思就是复制。在java中,克隆也是进行对象复制的操作。
how——怎样使用克隆?
实现克隆首先被克隆对象要实现Cloneable接口,同时要重写clone()方法,并且将方法权限设为public(需外不访问)!
why——为什么要克隆?
在开发中,多个地方共同引用、操作一个引用变量,但引用指向的物理存储堆是相同,操作之间会相互影响!因此,此时需要将操作引用对象进行克隆,避免多操作带来的相互影响!
下面就实现以下clone的使用:
public class CloneTest implements Cloneable {
private String element1;
public Object clone() {
CloneTest cloneTest = null;
try {
cloneTest = (CloneTest)super.clone();
}catch(CloneNotSupportedException e) {
e.printStackTrace();
}
return cloneTest;
}
public String getElement1() {
return element1;
}
public void setElement1(String element1) {
this.element1 = element1;
}
}
在重写实现clone方法的时,首先应保证其修饰权限为public,由于我们需要在类外引用该方法。接下来实现了Cloneable接口,我们点开Cloneable接口,发现里面并没有任何方法,为啥呢?先往下看,clone方法调用了Object类的clone方法,Object中clone是个本地方法,并且抛出CloneNotSupportedException异常,此时就理解上面的Cloneable空接口的作用——标记克隆类!
克隆又有深克隆浅克隆或叫深拷贝浅拷贝,个人认为深浅克隆没啥分别,只是我们使用用途的不同而需要的clone对象不同而已,浅克隆:未递归将克隆类中的引用变量克隆(未实现克隆),反之则是深克隆。下面代码展示一下深浅克隆。
public class CloneTest implements Cloneable {
private String element1;
private CloneObj cloneObj ;
public Object clone() {
CloneTest cloneTest = null;
try {
cloneTest = (CloneTest)super.clone();
}catch(CloneNotSupportedException e) {
e.printStackTrace();
}
return cloneTest;
}
public String getElement1() {
return element1;
}
public void setElement1(String element1) {
this.element1 = element1;
}
public CloneObj getCloneObj() {
return cloneObj;
}
public void setCloneObj(CloneObj cloneObj) {
this.cloneObj = cloneObj;
}
}
public class CloneObj {
private String elemOne;
public String getElemOne() {
return elemOne;
}
public void setElemOne(String elemOne) {
this.elemOne = elemOne;
}
}
上面代码为浅克隆,即在CloneTest类中引用的CloneObj也是克隆类,如果CloneObj未实现克隆,如下代码,则是深克隆。深浅克隆根据自己业务场景来处理自己的数据,个人觉得区不区分没意义!
public class CloneTest implements Cloneable {
private String element1;
private CloneObj cloneObj ;
public Object clone() {
CloneTest cloneTest = null;
try {
cloneTest = (CloneTest)super.clone();
cloneTest.cloneObj = (CloneObj) cloneObj.clone();
}catch(CloneNotSupportedException e) {
e.printStackTrace();
}
return cloneTest;
}
public String getElement1() {
return element1;
}
public void setElement1(String element1) {
this.element1 = element1;
}
public CloneObj getCloneObj() {
return cloneObj;
}
public void setCloneObj(CloneObj cloneObj) {
this.cloneObj = cloneObj;
}
}
public class CloneObj implements Cloneable {
private String elemOne;
public Object clone() {
CloneObj cloneObj = null;
try {
cloneObj = (CloneObj)super.clone();
}catch(CloneNotSupportedException e) {
e.printStackTrace();
}
return cloneObj;
}
public String getElemOne() {
return elemOne;
}
public void setElemOne(String elemOne) {
this.elemOne = elemOne;
}
}
总结:
克隆即是将对堆内的数据copy了一份,重新将指针分配其他变量,增大了内存的开销,因此不可滥用!!!