Java的浅拷贝和深拷贝

什么是Java的浅拷贝?

浅拷贝又称浅复制,浅克隆。浅拷贝是指拷贝时只拷贝对象本身(包括对象中的基本变量),而不拷贝对象包含的引用所指向的对象,拷贝出来的对象的所有变量的值都含有与原来对象相同的值,而所有对其他对象的引用都指向原来的对象,简单地说,浅拷贝只拷贝对象不拷贝引用。

什么是Java的深拷贝?

深拷贝又称为深复制,深克隆。深拷贝不仅拷贝对象本身,而且还拷贝对象包含的引用所指向的对象,拷贝出来的对象的所有变量(不包含那些引用其他对象的变量)的值都含有与原来对象的相同的值,那些引用其他对象的变量将指向新复制出来的新对象,而不指向原来的对象,简单地说,深拷贝不仅拷贝对象,而且还拷贝对象包含的引用所指向的对象。

再简单的说就是浅拷贝只拷贝对象,不拷贝引用。深拷贝对象引用全拷贝

拷贝的引入

引用拷贝

创建一个指向对象的引用变量的拷贝。

Student student = new Student("张三", 12);
Student copyStudent = student;
System.out.println(student);
System.out.println(copyStudent);

输出结果:

com.rjxy.copy.Student@1b6d3586
com.rjxy.copy.Student@1b6d3586

分析:从结果可以看出,它们输出的地址值是相同的,那么它们肯定是同一个对象。new Student("张三",12) 时在堆区创建了,而变量studentcopyStudent只是引用而已,它们都指向了同一个对象。这就叫做引用拷贝。

例1 图解:

image-20220724003033672

对象拷贝

创建对象本身的一个副本。

例2:

Student student = new Student("张三", 12);
Student cloneStudent = (Student) student.clone();
System.out.println(student);
System.out.println(cloneStudent);

输出结果

com.rjxy.copy.Student@1b6d3586
com.rjxy.copy.Student@4554617c

分析:由输出结果可以看出,它们的地址是不同的,也就是说创建了新的对象, 而不是把原对象的地址赋给了一个新的引用变量,这就叫做对象拷贝。

image-20220724003515997

注:深拷贝和浅拷贝都是对象拷贝

浅拷贝

浅拷贝仅仅复制所考虑的对象,而不复制它所引用的对象。

实现Cloneable重写clone

注:此方法默认是浅克隆

例3:

定义两个类TeacherStudent

Teacher类:

class Teacher implements Cloneable {
    String name;

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

    @Override
    public String toString() {
        return "Teacher{" +
            "name='" + name + '\'' +
            '}';
    }
}

Student类:

class Student implements Cloneable {
    String name;
    Integer id;
    Teacher teacher;

    public Student(String name, Integer id, Teacher teacher) {
        this.name = name;
        this.id = id;
        this.teacher = teacher;
    }

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

    @Override
    public String toString() {
        return "Student{" +
            "name='" + name + '\'' +
            ", id=" + id +
            ", teacher=" + teacher +
            '}';
    }
}

测试:

public class CopyTest {
        public static void main(String[] args) throws Exception {
            Teacher teacher = new Teacher("李四老师");
            Student student = new Student("张三", 12, teacher);
            Student cloneStudent = (Student) student.clone();
            System.out.println(student);
            System.out.println(cloneStudent);
            teacher.name = "王五老师";
            System.out.println("=========修改老师信息后=========");
            System.out.println(student);
            System.out.println(cloneStudent);
        }
    }
}

输出结果:

Student{name='张三', id=12, teacher=Teacher{name='李四老师'}}
Student{name='张三', id=12, teacher=Teacher{name='李四老师'}}
=========修改老师信息后=========
Student{name='张三', id=12, teacher=Teacher{name='王五老师'}}
Student{name='张三', id=12, teacher=Teacher{name='王五老师'}}

例3 图解:

image-20220724005719719

分析:

​ 两个studentcloneStudent指向不同的对象,但是两个引用studentcopyStudent中的teacher引用指向的是同一个对象,所以说是浅拷贝。

深拷贝

深拷贝是把要复制的对象所引用的对象都复制一遍。

例4:

Teacher类:

class Teacher implements Cloneable {
    String name;

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

    @Override
    public String toString() {
        return "Teacher{" +
            "name='" + name + '\'' +
            '}';
    }

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

Student类:

class Student implements Cloneable {
    String name;
    Integer id;
    Teacher teacher;

    public Student(String name, Integer id, Teacher teacher) {
        this.name = name;
        this.id = id;
        this.teacher = teacher;
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        // 浅复制时
        // return super.clone();

        //深复制时
        Student student = (Student) super.clone();
        student.teacher = (Teacher) student.teacher.clone();
        return student;
    }

    @Override
    public String toString() {
        return "Student{" +
            "name='" + name + '\'' +
            ", id=" + id +
            ", teacher=" + teacher +
            '}';
    }
}

测试代码:

Teacher teacher1 = new Teacher("李四老师");
Student student = new Student("张三", 12, teacher1);
Student cloneStudent = (Student) student.clone();
System.out.println(student);
System.out.println(cloneStudent);

teacher.name = "王五老师";
System.out.println("=========修改老师信息后=========");
System.out.println(student);
System.out.println(cloneStudent);

结果:

Student{name='张三', id=12, teacher=Teacher{name='李四老师'}}
Student{name='张三', id=12, teacher=Teacher{name='李四老师'}}
=========修改老师信息后=========
Student{name='张三', id=12, teacher=Teacher{name='王五老师'}}
Student{name='张三', id=12, teacher=Teacher{name='李四老师'}}

结果分析:

  两个引用studentcopyStudent 指向不同的对象,这两个对象中的成员变量teacher是指向的两个对象,对teacher1修改只能影响到student对象,所以说是深拷贝。

例4 图解(teacher姓名更改前)

image-20220724103345731

例4 图解(teacher姓名更改后)

image-20220724103431359

深拷贝的几种方式

1.clone()实现

2.构造函数实现

Teacher teacher = new Teacher("李四老师");
Student student = new Student("张三", 12, teacher);
Student cloneStudent = new Student("小二", 11, new Teacher(teacher.name));
System.out.println(student);
System.out.println(cloneStudent);

teacher.name = "王五老师";
System.out.println("=========修改老师信息后=========");
System.out.println(student);
System.out.println(cloneStudent);

3.序列化

public Object deepClone() throws Exception {
    // 序列化
    ByteArrayOutputStream bos = new ByteArrayOutputStream();
    ObjectOutputStream oos = new ObjectOutputStream(bos);
    oos.writeObject(this);

    // 反序列化
    ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
    ObjectInputStream ois = new ObjectInputStream(bis);
    return ois.readObject();
}

4.Apache的Commons-lang

注:对象需要实现Serializable接口

<dependency>
    <groupId>org.apache.commons</groupId>
    <artifactId>commons-lang3</artifactId>
    <version>3.1</version>
</dependency>
public Student deepClone2() {
    return SerializationUtils.clone(this);
}

5.Jackson的JSON序列化

注:使用Jackson的JSON序列化时要为被序列化类加上getset方法,以及无参构造

public Student deepClone3() throws Exception {
    ObjectMapper objectMapper = new ObjectMapper();
    return objectMapper.readValue(objectMapper.writeValueAsString(this), Student.class);
}

6.Gson的JSON序列化

7.FastJson的JSON序列化

💻参考自Java深入 深拷贝与浅拷贝详解

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值