浅拷贝和深拷贝的区别

java 复制代码
        Person p1 = new Person(10);
        Person p2 = p1;
        p2.age = 20;

        System.out.println(p1=p2); // true
        System.out.println(p1.age); // 20

这种做法只是复制了对象的地址,即两个变量现在是指向了同一个对象,任意一个变量,操作了对象的属性,都会影响到另一个变量

这中对同一个对象操作,当然算不上真正的复制,所以引用拷贝并不算对象拷贝,所谓的对象拷贝一般就是指浅拷贝和深拷贝

浅拷贝

在java中Object提供了一个clone()方法,看名字就是它和对象拷贝有关,该方法访问修饰符为protected,如果子类不重写该方法,并将其声明为public,那外部就调用不了,对象的clone().

java 复制代码
class Person implements Cloneable {
    public int age;

    public Person(int age) {
        this.age = age;
    }

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

子类在重写时直接调用Object的clone()即可,它是native方法,底层实现了拷贝对象的逻辑,注意子类一定得实现Cloneable接口,否则调用clone()时,会抛出异常,这是java的规定。

java 复制代码
        Person p1 = new Person(10);
        Person p2 = p1;
        p2.age = 20;

        System.out.println(p1=p2); // false
        System.out.println(p1.age); // 10

现在我们调用clone()方法来实现,发现两个变量指向的已经是不同的对象各自改属性,也不会影响到另一个对象,看起来效果很好。

不过有一个问题,如果拷贝对象中有属性是引用类型,那这种浅拷贝的方式,只会复制该属性的引用地址,即拷贝对象和原对象的属性,都指向了同一个对象,如果对这个属性进行一些操作,则会影响到另一个对象的属性,若想将对象中的引用类型属性也进行拷贝,那就得用深拷贝了。

深拷贝

java 复制代码
class Person implements Cloneable {
    public int age;
    public int[] arr = new int[]{1,2,3}

    public Person(int age) {
        this.age = age;
    }

    @Override
    public Person clone() {
        try {
           Person person = (Person) super.clone();  
           person.arr = this.arr.clone();
           return perosn;
        } catch (CloneNotSupportedException e) {
            throw new RuntimeException(e);
        }
    }
}

我们将clone()方法稍微修改一下,clone()出对象之后,我们再对对象的属性,进行一次拷贝。

这样就完成了属性的复制,此时对象中的属性也指向了不同的对象实例。

总结

引用拷贝只是复制对象的地址,并不会创建一个新的对象

浅拷贝会创建一个对象,并进行属性的复制,不过对引用类型的属性,只会复制其对象地址

深拷贝则是完全复制整个对象,包括引用类型的属性

都是通过clone(),在实际开发中,不建议大家使用该方法,因为他有抛出异常的风险,如果真的想让对象提供拷贝功能,可以自己编写其他方法来实现

相关推荐
似水明俊德4 小时前
02-C#.Net-反射-面试题
开发语言·面试·职场和发展·c#·.net
Leinwin4 小时前
OpenClaw 多 Agent 协作框架的并发限制与企业化规避方案痛点直击
java·运维·数据库
薛定谔的悦4 小时前
MQTT通信协议业务层实现的完整开发流程
java·后端·mqtt·struts
enjoy嚣士5 小时前
springboot之Exel工具类
java·spring boot·后端·easyexcel·excel工具类
Thera7775 小时前
C++ 高性能时间轮定时器:从单例设计到 Linux timerfd 深度优化
linux·开发语言·c++
罗超驿5 小时前
独立实现双向链表_LinkedList
java·数据结构·链表·linkedlist
炘爚5 小时前
C语言(文件操作)
c语言·开发语言
阿蒙Amon5 小时前
C#常用类库-详解SerialPort
开发语言·c#
盐水冰6 小时前
【烘焙坊项目】后端搭建(12) - 订单状态定时处理,来单提醒和顾客催单
java·后端·学习
凸头6 小时前
CompletableFuture 与 Future 对比与实战示例
java·开发语言