java面試基礎-Java中對Clone的理解

面試中經常遇到Clone的相關知識,今天總算是把Clone理解的比較透徹了!Java中Clone的概念大家應該都很熟悉了,它可以讓我們很方便的“製造”出一個對象的副本來,下面來具體看看java中的Clone機制是如何工作的?

1. Clone和Copy

 假設現在有一個User對象,User u1=new User(“U1001”,“Jason”,25),通

常我們會有這樣的賦值User u2=u1,這個時候只是簡單了copy了一下reference,u2和u1都指向內存中同一個object,這樣u2或者u1的一個操作都可能影響到對方。打個比方,如果我們通過u2.setAge()方法改變了Age域的值,那麼u1通過getAge()方法得到的就是修改之後的Age域的值,顯然這不是我們願意看到的。我們希望得到u1的一個精確拷貝,同時兩者互不影響,這時候我們就可以使用Clone來滿足我們的需求。User u2=u1.clone(),這時會生成一個新的User對象,並且和u1具有相同的屬性值和方法。

2. Shallow Clone和Deep Clone

Clone是如何完成的呢?Object在對某個對象實施Clone時對其是一無所知的,它僅僅是簡單地執行域對域的copy,這就是Shallow Clone。這樣,問題就來了咯,以User爲例,它裏面有一個域birthday不是基本型別的變量,而是一個reference變量,經過Clone之後就會產生一個新的Date型別的reference,它和原始對象中對應的域指向同一個Date對象,這樣克隆類就和原始類共享了一部分信息,而這樣顯然是不利的,過程下圖所示:

這個時候我們就需要進行deep Clone了,對那些非基本型別的域進行特殊的處理,例如本例中的birthday。我們可以重新定義Clone方法,對birthday做特殊處理,如下代碼所示:
Java代碼 收藏代碼

 class User implements Cloneable  
    {  
         public Object clone() throws CloneNotSupportedException  
         {  
           User cloned = User super.clone();  
           cloned.birthday = (Date) hireDay.clone()  
           return cloned;  
         }  
    }  

3. Clone()方法的保護機制在Object中Clone()是被申明爲protected的,這樣做是有一定的道理的,以User類爲例,通過申明爲protected,就可以保證只有User類裏面才能“克隆”User對象,原理可以參考我前面關於public、protected、private的學習筆記。

4. Clone()方法的使用Clone()方法的使用比較簡單,注意如下幾點即可:a. 什麼時候使用shallow Clone,什麼時候使用deep Clone,這個主要看具體對象的域是什麼性質的,基本型別還是reference variableb. 調用Clone()方法的對象所屬的類(Class)必須implements Clonable接口,否則在調用Clone方法的時候會拋出CloneNotSupportedException。

來一個深度的,不過這也不深,還可以克到五段,六段!

package com.colorme.t20.deepclone;
/**
* 被克隆對象裏包含的引用類
* @author lcc
* @version 2017-07-11
*/

public class ReferencedObject implements Cloneable{
    private int i = 0;
    public ReferencedObject(int i) {
        this.i = i;
    }
    public void doubleValue(){
        this.i = 2*this.i;
    }

    public int getNumber(){
        return this.i;
    }

public  Object clone() throws CloneNotSupportedException {
return super.clone();
}


}

package com.feixun.t20.deepclone;

/**
* 帶引用的克隆
* @author lcc
* @version 2017-07-11
*/
public class ObjectForDeepClone  implements Cloneable{
    private int i = 0;
    private ReferencedObject rf = null;

    public void setNum(int i) {
        this.i = i;
    }

    public void setReferencedObject(ReferencedObject rf){
        this.rf = rf;
    }

    public int getNumber(){
        return this.i;
    }

    public ReferencedObject getReferencedObject(){
        return this.rf;
    }

    // 重寫Object的clone
    public Object clone() throws CloneNotSupportedException {
    ObjectForDeepClone cloneObject=(ObjectForDeepClone)super.clone();
    // 調用引用對象的clone方法
        if(rf!=null){
            ReferencedObject rfClone = (ReferencedObject)rf.clone();
            cloneObject.setReferencedObject(rfClone);
        }

         return cloneObject;
     }
}  

package com.feixun.t20.deepclone;

import java.io.Serializable;

public class CloneTest {

    public static void main(String[] args) {
        ObjectForDeepClone ofsc = new ObjectForDeepClone();
        ofsc.setNum(888);
        ofsc.setReferencedObject(new ReferencedObject(1));

        // 開始clone了!
        ObjectForDeepClone deepCloneObject = null;
        try {
            deepCloneObject = (ObjectForDeepClone) ofsc.clone();
        } catch (CloneNotSupportedException ex) {
            System.out.println("Sorry,Clone Not Supported!");
        }

        // 測試clone是否成功
        if(deepCloneObject!=null){
            System.out.println("before clone");
            System.out.println("ObjectForShallowClone get number : " + ofsc.getNumber());
            System.out.println("ObjectForShallowClone ReferencedObject get number : "+ofsc.getReferencedObject().getNumber()+"\n");

            deepCloneObject.setNum(999);
            deepCloneObject.getReferencedObject().doubleValue();

            System.out.println("after clone");
            System.out.println("CloneObject get number : " + deepCloneObject.getNumber());
            System.out.println("CloneObject ReferencedObject get number : " + deepCloneObject.getReferencedObject().getNumber());

            System.out.println();
            System.out.println("ObjectForShallowClone get number : " + ofsc.getNumber());
            System.out.println("ObjectForShallowClone ReferencedObject get number : " + ofsc.getReferencedObject().getNumber());
        }      
    }
}

理解有誤之處,還望高 手 指 點!

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章