什么是克隆
克隆憾赁,顧名思義就是復(fù)制一個(gè)對(duì)象的的所有當(dāng)前狀態(tài)娱颊,并產(chǎn)生一個(gè)新的對(duì)象构挤,該對(duì)象與被復(fù)制的對(duì)象具有相同的狀態(tài)髓介。
怎樣實(shí)現(xiàn)克隆
如果一個(gè)類的實(shí)例想要實(shí)現(xiàn)克隆的功能,那么只需要該類實(shí)現(xiàn)Cloneable這個(gè)標(biāo)記接口(標(biāo)記接口是指沒(méi)有任何方法筋现,只有一個(gè)接口聲明的接口唐础,向JVM聲明該類實(shí)現(xiàn)了某種功能),并覆寫(xiě)繼承自O(shè)bject的clone方法即可矾飞。
如何實(shí)現(xiàn)對(duì)象克隆
當(dāng)一個(gè)類實(shí)現(xiàn)了Cloneable這個(gè)標(biāo)記接口并覆寫(xiě)了繼承自O(shè)bject的clone方法時(shí)一膨,并將限定修飾符修改為public,就可以實(shí)現(xiàn)對(duì)象克隆。在Object類中洒沦,clone方法為protected豹绪,為了保證可以在自己的業(yè)務(wù)邏輯處理中使用克隆方法,需要將方法的限定修飾符修改為public微谓。在Object類中森篷,該方法會(huì)拋出一個(gè) CloneNotSupportedException異常输钩,可以選擇把這個(gè)異常拋出交給業(yè)務(wù)處理代碼去處理豺型,也可以在覆寫(xiě)的clone方法中去捕獲仲智。
示例代碼如下
public class Person implements Cloneable {
@Override
public Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
深克與淺克隆的問(wèn)題怎么產(chǎn)生的
探究這個(gè)問(wèn)題需要了解一個(gè)實(shí)現(xiàn)了Cloneable接口并覆寫(xiě)了clone方法的類是如何實(shí)現(xiàn)克隆的。如上示例代碼所示姻氨,實(shí)現(xiàn)克隆就是調(diào)用Object類的克隆方法钓辆,即super.clone()。在Object類中肴焊,克隆方法是一個(gè)本地方法前联,它實(shí)現(xiàn)了對(duì)象克隆的所有操作。當(dāng)你調(diào)用對(duì)象實(shí)例的clone()時(shí)娶眷,實(shí)際時(shí)通過(guò)Object的本地方法類來(lái)完成的似嗤。問(wèn)題就是在這產(chǎn)生的。由于Object是所有Java類的超類届宠,它并不了解子類的實(shí)現(xiàn)細(xì)節(jié)烁落,只能去遍歷子類的每個(gè)實(shí)例域,如果子類的實(shí)例域都是基本類型或是不可變類的話豌注,克隆出的對(duì)象的實(shí)例域和原有對(duì)象實(shí)例域同時(shí)指向同一個(gè)對(duì)象(實(shí)際就是內(nèi)存中的同一塊地址)伤塌,由于實(shí)例域都是基本類型或不可變類,那么克隆對(duì)象和原對(duì)象并不會(huì)產(chǎn)生問(wèn)題轧铁,大家的狀態(tài)都不會(huì)改變每聪,所以并不會(huì)產(chǎn)生問(wèn)題。但是實(shí)際開(kāi)發(fā)中這情況是非常少見(jiàn)的齿风,我們?nèi)粘i_(kāi)發(fā)中類的實(shí)例域往往都是可變的药薯,那么問(wèn)題就來(lái)了,實(shí)例域是可變的救斑,那么克隆出的對(duì)象和原對(duì)象的實(shí)例域指向同一個(gè)對(duì)象鐵定出問(wèn)題果善,一個(gè)對(duì)象的實(shí)例域的修改都會(huì)影響到另一個(gè)對(duì)象的狀態(tài),而我們往往希望克隆出的對(duì)象和原對(duì)象誰(shuí)都不影響誰(shuí)系谐,大家 各走各路巾陕,這種情況就是我們常說(shuō)的淺拷貝。
public class Salary {
private double baseSalay;
private double bonus;
// 省略get set方法
}
public class Person implements Cloneable {
private String height;
private Salary salary;
// 省略get set方法
@Override
public Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
如上代碼纪他,一個(gè)Person類有一個(gè)String類型的height屬性和一個(gè)Salary類型的salary屬性鄙煤,且Person類覆寫(xiě)了超類的clone方法,我們來(lái)看一下下面兩行代碼
Person peter=new Person();
Person harry=perter.clone();
原對(duì)象peter和克隆對(duì)象harry的實(shí)例域指向了一個(gè)String類型的height引用和一個(gè)Salary類型的salary引用茶袒,String類型是不可變的梯刚,這是ok的,克隆出來(lái)的有一樣的身高無(wú)可厚非,但世界上沒(méi)有相同的兩片葉子薪寓,任何個(gè)體都有其特性亡资,假設(shè)harry機(jī)遇更好一點(diǎn)澜共,薪資更高,我們?cè)囍鴣?lái)為harry給更高的薪資锥腻,然后打印出peter和harry,你會(huì)神奇的發(fā)現(xiàn)peter的薪資竟然也提高了并且和harry一樣嗦董,這是因?yàn)樯厦娴拇a只是實(shí)現(xiàn)了對(duì)象的淺拷貝造成的,那么我們?nèi)绾巫宲eter和harry互不影響瘦黑,成為獨(dú)立的兩個(gè)個(gè)體呢京革?你需要考慮深拷貝了。
深拷貝是當(dāng)一個(gè)類中存在可變的實(shí)例域時(shí)幸斥,該類的clone方法中也要相應(yīng)的克隆該實(shí)例域并復(fù)制給克隆對(duì)象匹摇。下面來(lái)更改一下代碼實(shí)現(xiàn)。
public class Salary implements Cloneable{
private double baseSalay;
private double bonus;
// 省略get set方法
@Override
protected Salary clone() throws CloneNotSupportedException {
return (Salary)super.clone();
}
}
public class Person implements Cloneable {
private String height;
private Salary salary;
// 省略get set方法
@Override
public Person clone() throws CloneNotSupportedException {
Person clone = (Person) super.clone();
clone.salary = salary.clone();
return clone;
}
}
Salary類實(shí)現(xiàn)了Cloneable接口并覆寫(xiě)了clone方法甲葬,在Person類的clone方法中調(diào)用salary的克隆方法并賦值給克隆出的對(duì)象的salary域廊勃,此時(shí),perter和harry終于可以不用互相影響了经窖,至此深拷貝就完成了坡垫。
考慮一下,如果Salary類不只是基本類型的實(shí)例域钠至,該怎么做呢葛虐?我們應(yīng)該在Salary類的clone方法繼續(xù)調(diào)用該類的可變實(shí)例域的clone方法,如果想要實(shí)現(xiàn)深拷貝棉钧,我們一定要保證類中不存在可變的實(shí)例域?qū)ο笥炱辏蝗恢荒苓f歸的確保每一個(gè)實(shí)例域都實(shí)現(xiàn)了clone方法。
總結(jié)
關(guān)于深拷貝與淺拷貝的坑這里還并沒(méi)有說(shuō)完宪卿,比如我們應(yīng)不應(yīng)該讓一個(gè)類覆寫(xiě)clone方法的诵,是否可以以更好的類設(shè)計(jì)來(lái)避免clone,這些都需要日常中多總結(jié)佑钾,思考西疤,學(xué)習(xí)。
路漫漫其修遠(yuǎn)兮休溶,吾將上下而求索代赁。。兽掰。