深拷貝與淺拷貝

什么是克隆

克隆憾赁,顧名思義就是復(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)兮休溶,吾將上下而求索代赁。。兽掰。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末芭碍,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子孽尽,更是在濱河造成了極大的恐慌窖壕,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,839評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異瞻讽,居然都是意外死亡鸳吸,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,543評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門(mén)速勇,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)晌砾,“玉大人,你說(shuō)我怎么就攤上這事快集」备幔” “怎么了廉白?”我有些...
    開(kāi)封第一講書(shū)人閱讀 153,116評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵个初,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我猴蹂,道長(zhǎng)院溺,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,371評(píng)論 1 279
  • 正文 為了忘掉前任磅轻,我火速辦了婚禮珍逸,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘聋溜。我一直安慰自己谆膳,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,384評(píng)論 5 374
  • 文/花漫 我一把揭開(kāi)白布撮躁。 她就那樣靜靜地躺著漱病,像睡著了一般。 火紅的嫁衣襯著肌膚如雪把曼。 梳的紋絲不亂的頭發(fā)上杨帽,一...
    開(kāi)封第一講書(shū)人閱讀 49,111評(píng)論 1 285
  • 那天,我揣著相機(jī)與錄音嗤军,去河邊找鬼注盈。 笑死,一個(gè)胖子當(dāng)著我的面吹牛叙赚,可吹牛的內(nèi)容都是我干的老客。 我是一名探鬼主播,決...
    沈念sama閱讀 38,416評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼震叮,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼胧砰!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起冤荆,我...
    開(kāi)封第一講書(shū)人閱讀 37,053評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤朴则,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體乌妒,經(jīng)...
    沈念sama閱讀 43,558評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡汹想,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,007評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了撤蚊。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片古掏。...
    茶點(diǎn)故事閱讀 38,117評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖侦啸,靈堂內(nèi)的尸體忽然破棺而出槽唾,到底是詐尸還是另有隱情,我是刑警寧澤光涂,帶...
    沈念sama閱讀 33,756評(píng)論 4 324
  • 正文 年R本政府宣布庞萍,位于F島的核電站,受9級(jí)特大地震影響忘闻,放射性物質(zhì)發(fā)生泄漏钝计。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,324評(píng)論 3 307
  • 文/蒙蒙 一齐佳、第九天 我趴在偏房一處隱蔽的房頂上張望私恬。 院中可真熱鬧,春花似錦炼吴、人聲如沸本鸣。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,315評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)荣德。三九已至,卻和暖如春提针,著一層夾襖步出監(jiān)牢的瞬間命爬,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,539評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工辐脖, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留饲宛,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,578評(píng)論 2 355
  • 正文 我出身青樓嗜价,卻偏偏與公主長(zhǎng)得像艇抠,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子久锥,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,877評(píng)論 2 345