Java中的深拷贝与浅拷贝

前言

这段时间在看《设计模式之禅》这本书,其中讲到了原型模式,可说老实话在此之前我从来没听说过还有这种模式的存在,是我孤陋寡闻了!在阅读了相关文章后,引出了深拷贝和浅拷贝,之前了解过这个知识点的,到现在完全没有印象了。就此,做出如下简述。

正文

在说深拷贝和浅拷贝之前先了解Java中的Cloneable接口

在这里插入图片描述

该接口与java.util.RandomAccess接口一样,仅起标识作用。实现此接口并重写clone()方法,即可实现类的拷贝功能。不管是深拷贝(非必须)还是浅拷贝都依赖于此接口。

浅拷贝

浅拷贝(Shallow Clone)从其名字大致可以了解其特性。在拷贝过程中,对于不同类型类型做出了区分,

  • 对于基本数据类型拷贝值,进行值传递
  • 对于引用数据类型只是进行了引用传递

通过实例代码查看执行结果:

@ToString
public class ShallowClone implements Cloneable {
    private int age;
    private List<String> list = new ArrayList<>();

    public ShallowClone() {
    }

    public ShallowClone(int age, List<String> list) {
        this.age = age;
        this.list = list;
    }

    public int getAge() {
        return age;
    }

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

    public List<String> getList() {
        return list;
    }

    public void setList(String str) {
        this.list.add(str);
    }

    @Override
    protected ShallowClone clone() throws CloneNotSupportedException {
        return (ShallowClone) super.clone();
    }

    public static void main(String[] args) throws CloneNotSupportedException {
        // 为shallowClone赋值
        ShallowClone shallowClone = new ShallowClone();
        shallowClone.setAge(12);
        shallowClone.setList("123");

        // clone后对新对象进行赋值
        ShallowClone clone = shallowClone.clone();
        // 如果新对象不设置age属性,age还是12
        clone.setAge(11);
        // 因为对引用类型是不进行拷贝,所以对list的两次添加数据会保存在一起
        clone.setList("456");

        // ShallowClone(age=12, list=[123, 456])
        System.out.println(shallowClone.toString());
        // ShallowClone(age=11, list=[123, 456])
        System.out.println(clone.toString());
    }
}

拷贝过程如下图所示

在调用了clone()方法后再赋值,对于age字段来说并没有影响,但对于list就相当于执行了两次add()操作。

深拷贝

深拷贝(Deep Clone)对比于前者的区别在于对引用类型的复制,会创建一个新对象,并复制其内容。这里使用Hutool中的深拷贝方法ObjectUtil.cloneByStream(),实现Serializable接口即可,通过序列化完成拷贝

@ToString
public class DeepClone implements Serializable {
    private static final long serialVersionUID = -174641130567660496L;
    private int age;
    private List<String> list = new ArrayList<>();

    public DeepClone() {
    }

    public DeepClone(int age, List<String> list) {
        this.age = age;
        this.list = list;
    }

    public int getAge() {
        return age;
    }

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

    public List<String> getList() {
        return list;
    }

    public void setList(String str) {
        this.list.add(str);
    }

    public static void main(String[] args) throws CloneNotSupportedException {
        // 初始化原对象
        DeepClone deepClone = new DeepClone();
        deepClone.setAge(12);
        deepClone.setList("123");

        // 执行深拷贝
        DeepClone clone = ObjectUtil.cloneByStream(deepClone);
        clone.setAge(11);
        clone.setList("456");
        clone.setList("456");
        clone.setList("456");

        // DeepClone(age=12, list=[123])
        System.out.println(deepClone.toString());
        // DeepClone(age=11, list=[123, 456, 456, 456])   对clone的改变并没有影响deepClone
        System.out.println(clone.toString());
    }
}

这里使用HuTool中的深拷贝方法ObjectUtil.cloneByStream(),其底层的执行逻辑如下图

阅读原文

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值