Java 中的深拷贝和浅拷贝

概念引入:引用拷贝
二者的引用是同一个对象,并没有创建出一个新的对象
因为是同一个对象的引用,所以两者改一个,另一个对象的值也随之改变。
在这里插入图片描述
引用拷贝包括浅拷贝与深拷贝

浅拷贝(shallowCopy):复制基本类型的属性;引用类型的属性复制,复制栈中的变量 和 变量指向堆内存中的对象的指针,不复制堆内存中的对象;
深拷贝(deepCopy):复制基本类型的属性;引用类型的属性复制,复制栈中的变量 和 变量指向堆内存中的对象的指针和堆内存中的对象
在这里插入图片描述
在这里插入图片描述
浅拷贝测试代码:

class Teacher implements Cloneable{
   String name;

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

    public Teacher(String name){this.name=name;}

    @Override
    protected Object clone() throws CloneNotSupportedException {
        Teacher t1=(Teacher)super.clone();
        return t1;
    }
}
class Student implements Cloneable{
    public String name;
    public Teacher teacher;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }

    public Teacher getTeacher() {
        return teacher;
    }
    public void setTeacher(Teacher teacher) {
        this.teacher = teacher;
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        Student s1=(Student)super.clone();
        return s1;
    }
}
 @Test
    public void test4() throws CloneNotSupportedException {
      //初始化
        Teacher teacher1=new Teacher("teacher");
        Student student1=new Student();
        student1.setName("studenet1");
        student1.setTeacher(teacher1);
        //克隆
        Student student2=(Student)student1.clone();
        System.out.println(student1);
        System.out.println(student2);
        System.out.println("修改前student1.name:"+student1.getName());
        System.out.println("修改前student2.name:"+student2.getName());
        //修改student2.name
        student2.setName("student2");
        System.out.println("修改后student1.name:"+student1.getName());
        System.out.println("修改后student2.name:"+student2.getName());
        System.out.println(student1.getTeacher());
        System.out.println(student2.getTeacher());
    }

测试结果:
在这里插入图片描述
结果分析:
student1.clone()以后,student1与student2的地址不一样,修改student1.name,student2.name不会改变,这不就是深拷贝了吗?其实不是,引用student1对teacher的引用与student2对teacher的引用地址是一样的!?
在student类定义时,实现了Cloneable接口,并重写了Clone方法

@Override
    protected Object clone() throws CloneNotSupportedException {
        Student s1=(Student)super.clone();
        return s1;
    }

我们只对student这个类进行了clone,stuednt在内存中会有2份,可是为什么teacher只有一份?
带着这个疑问我们对student重写clone方法进行修改
深拷贝测试代码

@Override
    protected Object clone() throws CloneNotSupportedException {
        Student s1=(Student)super.clone();
        s1.teacher=(Teacher)this.teacher.clone();
        return s1;
    }
```java

这么做就要在super.clone的基础上 继续对非基本类型的对象递归的再次clone.

测试结果:
在这里插入图片描述
stduent1与stduent2的地址不同,内存中有2份,stuent1.teacher与student2.teacher的地址不同,内存中也有2份,修改student1.name不影响stduent2.name,修改student.teacher.name不影响stduent2.teacher.name,实现了深拷贝,需要在super.clone的基础上 继续对非基本类型的对象递归的再次clone

在java中,我们经常会将一个对象赋值给另一个对象,比如

Student s1 = new Studnet(12,"李四");
Student s2 = s1;
s2.setID(36); 
System.out.println(s1+"___"+s2);

当你对对象进行操作,不管是赋值之前还是之后,两个值都会修改,这是因为它们是指向同一个对象,指针不一样。

所以这就涉及深拷贝与浅拷贝,深拷贝就是复制一个对象到一个全新的对象,和之前的对象没有关联;

浅拷贝就是前面提到的直接赋值给另一个对象。

如果想实现深拷贝,就需要自己去实现,比如:

public class Student(){
    String name;
    int id;
    
    public Student deepClone(){
        Student s = new Student();
        s.id = this.id;
        ...
        return s;
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java深拷贝(Deep Copy)和浅拷贝(Shallow Copy)是用于复制对象的两种不同方式。 浅拷贝是指创建一个新对象,然后将原始对象的非静态字段的值复制到新对象。如果字段是基本数据类型,则复制其值;如果字段是引用类型,则复制引用而不是实际对象。这意味着新旧对象将共享同一个引用对象。当修改其一个对象的引用对象时,另一个对象也会受到影响。 深拷贝是指创建一个新对象,并将原始对象的所有字段的值复制到新对象,包括引用类型字段。这意味着新对象将拥有原始对象的副本,而不是共享引用。即使修改其一个对象的引用对象,另一个对象也不会受到影响。 Java提供了几种方式实现深拷贝浅拷贝: 1. 对于浅拷贝,可以使用`clone()`方法。该方法是Object类的一个protected方法,需要在要拷贝的类实现Cloneable接口,并重写`clone()`方法。然后通过调用`clone()`方法来创建拷贝对象。 2. 对于深拷贝,可以使用序列化和反序列化来实现。将对象写入输出流,然后再从输入流读取对象,可以得到一个全新的拷贝对象。 需要注意的是,如果原始对象包含引用类型的字段,那么引用类型对象也需要实现Cloneable接口并重写`clone()`方法,或者是可序列化的。否则,在进行浅拷贝深拷贝时,引用类型对象仍然会被共享或不被复制。 希望能解决你的问题!如果还有其他问题,请随时提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值