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
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值