淺談Java中的克隆機制


轉 https://blog.csdn.net/zhaoheng314/article/details/81985880

 

Java語言中克隆針對的是類的實例

/Library/Java/JavaVirtualMachines/jdk1.8.0_221.jdk/Contents/Home/jre/lib/rt.jar!/java/lang/Cloneable.class

根據Cloneable接口的注釋, 克隆需要遵循以下規則:

  1. 必須實現Cloneable接口
  2. 實現Cloneable的類應該重寫clone(),重寫時該方法的修飾符為public。

  

/Library/Java/JavaVirtualMachines/jdk1.8.0_221.jdk/Contents/Home/jre/lib/rt.jar!/java/lang/Object.class

protected native Object clone() throws CloneNotSupportedException;


age屬性是int類型,基本類型克隆時只是傳值,不存在傳引用
String類 final關鍵字 不可變特性,克隆對象修改了name屬性的引用


深克隆
public class Student implements Cloneable{

    private String name;//引用類型  final關鍵字 一是不可變,二是禁止指令重排

    private int age;//基本類型

    private Classes classes;//引用類型

    public void setName(String name){
        this.name = name;
    }

    public void setAge(int age){
        this.age = age;
    }

    public void setClasses(Classes classes){
        this.classes = classes;
    }

    public String getName(){
        return name;
    }

    public Classes getClasses(){
        return classes;
    }

    @Override
    public String toString() {
        return "Student [name=" + name + ", age=" + age + ", classes=" + classes + "]";
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        Student stu = (Student)super.clone();
        //克隆classes屬性時候調用Classes類的clone();
        Classes cla = (Classes)classes.clone();
        stu.setClasses(cla);
        return stu;
    }
}

 

public class Classes implements Cloneable{

    private int classId;//基本類型

    private String className;//引用類型

    public void setClassId(int classId){
        this.classId = classId;
    }

    public void  setClassName(String className){
        this.className = className;
    }


    @Override
    public String toString() {
        return "Classes [classId=" + classId + ", className=" + className + "]";
    }

    @Override
    protected Object clone() throws CloneNotSupportedException{
        return super.clone();
    }
}
public class TestClone {
    public static void main(String[] args) {
        Student stu = new Student();
        stu.setName("張三");
        stu.setAge(10);

        Classes classes = new Classes();
        classes.setClassId(101);
        classes.setClassName("一班");
        stu.setClasses(classes);


        try{
            System.out.println("深克隆測試-----");
            //克隆
            Student stu2 = (Student)stu.clone();
            System.out.println("兩個對象是否相同:" + (stu == stu2));
            System.out.println("兩個對象name屬性是否相同:" + (stu.getName() == stu2.getName()));
            System.out.println("兩個對象classes是否相同:" + (stu.getClasses() == stu2.getClasses()));
            System.out.println("深克隆,Stu" + stu);
            System.out.println("深克隆,Stu2" + stu2);

            System.out.println("修改克隆對象屬性");
            stu2.setName("李四"); // public final class String 一是不可變,二是禁止指令重排 修改了name屬性的引用
            stu2.setAge(20);
            stu2.getClasses().setClassId(102);
            stu2.getClasses().setClassName("二班");
            System.out.println("修改克隆對象屬性后,Stu " + stu);
            System.out.println("修改克隆對象屬性后,Stu2 " + stu2);

        }catch (CloneNotSupportedException e){
            e.printStackTrace();
        }



    }
}

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM