引用拷贝、浅拷贝、深拷贝

引用拷贝

@Data
class Father {
    String name;
    Integer age;
    Son son;

    Father(String name, Integer age, Son son) {
        this.name = name;
        this.age = age;
        this.son = son;
    }
}

@Data
class Son {
    String name;

    Son(String name) {
        this.name = name;
    }
}


public class CloneTest {
    public static void main(String[] args) {
        Father father = new Father("father", 25, new Son("son"));
        Father copyFather = father;
        copyFather.name="copyName";
        copyFather.age = 85;
        copyFather.son.name="copySon";
        System.out.println(father);
        System.out.println(copyFather);
    }
}

Father(name=copyName, age=85, son=Son(name=copySon))
Father(name=copyName, age=85, son=Son(name=copySon))

浅拷贝

实现 Cloneable 接口,重写clone方法。

@Data
class Father implements Cloneable{
    String name;
    int age;
    Son son;

    Father(String name, int age, Son son) {
        this.name = name;
        this.age = age;
        this.son = son;
    }

    @Override
    protected Object clone(){
        try {
            return super.clone();
        } catch (CloneNotSupportedException e) {
            throw new RuntimeException(e);
        }
    }
}

@Data
class Son {
    String name;

    Son(String name) {
        this.name = name;
    }
}


public class CloneTest {
    public static void main(String[] args) {
        Father father = new Father("father", 25, new Son("son"));
        Father copyFather = (Father) father.clone();
        copyFather.name="copyName";
        copyFather.age = 85;
        copyFather.son.name="copySon";
        System.out.println(father);
        System.out.println(copyFather);
    }
}

Father(name=father, age=25, son=Son(name=copySon))
Father(name=copyName, age=85, son=Son(name=copySon))

你可以发现 son 不是深拷贝。

JAVA为什么String,Integer等对象在浅克隆时会被克隆,而其它Object不行?
String,Integer 这种不可变类在 深拷贝 中也是特例。如果你实现了深拷贝,你会发现这两类成员依然不会创建克隆,而是在深拷贝时传递引用。看似这操作违反了Java的深拷贝定义,但它们其实完全不需要也不能被深拷贝。最主要的原因是:String,Integer等包装类本没有实现Cloneable接口,故根本无法克隆,只能传递引用。我们自己重写clone方法时,对于String、Integer这些成员也不必去理会。—— JasperFang

深拷贝

@Data
class Father implements Cloneable{
    String name;
    int age;
    Son son;

    Father(String name, int age, Son son) {
        this.name = name;
        this.age = age;
        this.son = son;
    }

    @Override
    protected Object clone(){
        try {
            Father father= (Father) super.clone();
            father.setSon((Son) father.getSon().clone());
            return father;
        } catch (CloneNotSupportedException e) {
            throw new RuntimeException(e);
        }
    }
}

@Data
class Son implements Cloneable{
    String name;

    Son(String name) {
        this.name = name;
    }

    @Override
    protected Object clone() {
        try {
            return super.clone();
        } catch (CloneNotSupportedException e) {
            throw new RuntimeException(e);
        }
    }
}

public class CloneTest {
    public static void main(String[] args) {
        Son son=new Son("son");
        Father father = new Father("father", 25, son);

        Father copyFather = (Father) father.clone();
        copyFather.name="copyName";
        copyFather.age = 85;
        copyFather.son.name="copySon";
        System.out.println(father);
        System.out.println(copyFather);
    }
}

Father(name=father, age=25, son=Son(name=son))
Father(name=copyName, age=85, son=Son(name=copySon))

注:还有一种深拷贝是序列化。

@Data
class Father implements Serializable {
    String name;
    int age;
    Son son;

    Father(String name, int age, Son son) {
        this.name = name;
        this.age = age;
        this.son = son;
    }
}

@Data
class Son implements Serializable {
    String name;

    Son(String name) {
        this.name = name;
    }
}

public class CloneTest {
    public static void main(String[] args) throws IOException, ClassNotFoundException {
        Son son = new Son("son");
        Father father = new Father("father", 25, son);

        ObjectOutputStream outputStream = new ObjectOutputStream(new FileOutputStream("D:\\a.txt"));
        outputStream.writeObject(father);
        outputStream.close();

        ObjectInputStream inputStream = new ObjectInputStream(new FileInputStream("D:\\a.txt"));
        Father copyFather = (Father) inputStream.readObject();
        System.out.println(copyFather);
        copyFather.name = "copyName";
        copyFather.age = 85;
        copyFather.son.name = "copySon";
        System.out.println(copyFather);
    }
}

Father(name=father, age=25, son=Son(name=son))
Father(name=copyName, age=85, son=Son(name=copySon))

posted @   帅气的涛啊  阅读(12)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· DeepSeek 开源周回顾「GitHub 热点速览」
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· 单线程的Redis速度为什么快?
历史上的今天:
2022-05-12 关于xml文件解析时'&'不能被解析的问题
点击右上角即可分享
微信分享提示

目录