Clone()方法深度剖析

拷貝,顧名思義就是復(fù)制,分配新的內(nèi)存,產(chǎn)生新的對(duì)象,與new關(guān)鍵字類似。然而,拷貝中存在部分隱匿的差別,需要我們細(xì)細(xì)查看

// 學(xué)士:姓名 年齡 信息
public class Bachelor {
    private String name;
    private int age;
    private Info info;
    
    public Bachelor(String name,int age,Info info){
        this.name = name;
        this.age = age;
        this.info = info;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public Info getInfo() {
        return info;
    }
    public void setInfo(Info info) {
        this.info = info;
    }   
}
// 信息:技能 項(xiàng)目 
class Info{
    private String level;
    private Project project;
    
    public Info(String level,Project project){
        this.level = level;
        this.project = project;
    }
    public Project getProject() {
        return project;
    }
    public void setProject(Project project) {
        this.project = project;
    }
}
// 項(xiàng)目:項(xiàng)目名稱
class Project{
    private String projectName;
    
    public Project(String projectName){
        this.projectName = projectName;
    }
}
  1. 引用拷貝
    不同引用指向相同地址,并不分配新的內(nèi)存空間;
    // 測試代碼
    Project project = new Project("myApp");
    Info info = new Info("A",project);
    Bachelor bachelorOne = new Bachelor("Cheng",23,info);
    
    Bachelor bachelorTwo = bachelorOne;
    System.out.println(bachelorOne);
    System.out.println(bachelorTwo);
// 打印出相同的地址
javaBasic.Bachelor@15db9742
javaBasic.Bachelor@15db9742
  1. 對(duì)象拷貝
// 實(shí)現(xiàn)Cloneable接口,復(fù)寫clone()方法
public class Bachelor implements Cloneable{
    @Override
    protected Object clone() throws CloneNotSupportedException {
        Bachelor newBachelor = (Bachelor) super.clone();
        return newBachelor;
    }
}
    // 測試代碼
    Project project = new Project("myApp");
    Info info = new Info("A",project);
    Bachelor bachelorOne = new Bachelor("Cheng",23,info);

    Bachelor bachelorTwo = (Bachelor) bachelorOne.clone();
    System.out.println(bachelorOne);
    System.out.println(bachelorTwo);
    System.out.println(bachelorOne.getName() == bachelorTwo.getName());
// 打印出不同地址,且name引用相同
javaBasic.Bachelor@15db9742
javaBasic.Bachelor@6d06d69c
true
  • 淺拷貝
    如上面測試代碼,分配新的內(nèi)存,創(chuàng)建新對(duì)象,但是name屬性卻存在相同的引用,即拷貝主對(duì)象,卻不拷貝主對(duì)象里面的對(duì)象;
    如何實(shí)現(xiàn)name屬性的拷貝呢?
    // 重新生成name對(duì)象
    @Override
    protected Object clone() throws CloneNotSupportedException {
        Bachelor newBachelor = (Bachelor) super.clone();
        newBachelor.name = new String(this.name);
        return newBachelor;
    }

按照此種邏輯,info對(duì)象也是淺拷貝,如何實(shí)現(xiàn)主對(duì)象中對(duì)象的拷貝呢?

// 類實(shí)現(xiàn)Cloneable接口,復(fù)寫clone方法
// 在clone()方法中,實(shí)現(xiàn)主對(duì)象內(nèi)對(duì)象的重新生成
public class Bachelor implements Cloneable{
    @Override
    protected Object clone() throws CloneNotSupportedException {
        Bachelor newBachelor = (Bachelor) super.clone();
        newBachelor.name = new String(this.name);
        newBachelor.info = (Info) this.info.clone();
        return newBachelor;
    }
}

class Info implements Cloneable{
    @Override
    protected Object clone() throws CloneNotSupportedException {
        Info newInfo = (Info) super.clone();
        // 將level屬性和project方法均重新生成
        newInfo.level = new String(this.level);
        newInfo.project = (Project) this.project.clone();
        return newInfo;
    }

class Project implements Cloneable{
    @Override
    protected Object clone() throws CloneNotSupportedException {
        // TODO Auto-generated method stub
        return (Project) super.clone();
    }
}
// 測試代碼
    Project project = new Project("myApp");
    Info info = new Info("A",project);
    Bachelor bachelorOne = new Bachelor("Cheng",23,info);
        
    Bachelor bachelorTwo = (Bachelor) bachelorOne.clone();
    System.out.println(bachelorOne);
    System.out.println(bachelorTwo);
    System.out.println(bachelorOne.getName() == bachelorTwo.getName());
    System.out.println(bachelorTwo.getInfo() == bachelorOne.getInfo());
    System.out.println(bachelorTwo.getInfo().getProject() == bachelorOne.getInfo().getProject());
// 實(shí)現(xiàn)主對(duì)象內(nèi)部所有對(duì)象的拷貝
javaBasic.Bachelor@15db9742
javaBasic.Bachelor@6d06d69c
false
false
false

  • 深拷貝
    實(shí)現(xiàn)主對(duì)象全部結(jié)構(gòu)的拷貝,包括主對(duì)象中的對(duì)象;
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 內(nèi)存管理 簡述OC中內(nèi)存管理機(jī)制。與retain配對(duì)使用的方法是dealloc還是release,為什么?需要與a...
    丶逐漸閱讀 2,081評(píng)論 1 16
  • *面試心聲:其實(shí)這些題本人都沒怎么背,但是在上海 兩周半 面了大約10家 收到差不多3個(gè)offer,總結(jié)起來就是把...
    Dove_iOS閱讀 27,624評(píng)論 30 472
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法,類相關(guān)的語法,內(nèi)部類的語法,繼承相關(guān)的語法,異常的語法,線程的語...
    子非魚_t_閱讀 34,728評(píng)論 18 399
  • 今天早晨思雨老師分享了一篇文章《你怎么總是有那么多“但是”???》,看了此文,思緒萬千,這不正是自己的影子嗎...
    sunshine_fa47閱讀 602評(píng)論 0 0
  • 布施,看起來是利人,實(shí)際上是鋪設(shè)自己的財(cái)富之道。 持戒,看起來是束縛,實(shí)際上是造就自己的成功之道。 忍辱,看起來是...
    圣鳳閱讀 168評(píng)論 0 0

友情鏈接更多精彩內(nèi)容