详解深浅拷贝:从原理到实现的完整指南

目录

深拷贝和浅拷贝的区别?

实现深拷贝的三种方法是什么?

实现Cloneable接口并重写里面的clone()方法:

序列化和反序列化,实现Serializable接口

手动递归克隆


深拷贝和浅拷贝的区别?

浅拷贝在复制非引用字段时 整体复制,复制引用字段时 只复制源对象存储的引用对象的内存存储地址,不会复制一份新的内存内容

深拷贝在复制时不论引用字段还是非引用字段,全体都再复制一份

举例说明:

cs 复制代码
class Person
{
    // 值类型
    public int Age;
    // 引用类型(字符串/数组/类都是)
    public string Name;
}

// 原对象
Person p1 = new Person { Age = 18, Name = "张三" };
// 浅拷贝 → 创建了新对象 p2
Person p2 = (Person)p1.MemberwiseClone(); 

此时p2中复制的是Age = 18;

还有Name这个变量所对应的内存地址,没有再复制一个张三

此时内存里只有一个张三,两个对象的 Name 指向同一块内存
p1 (新对象) Age → 18(复制了新值) Name → 内存地址 A → "张三" p2 (另一个新对象) Age → 18(复制了新值) Name → 内存地址 A → 【和 p1 共用同一份字符串】
如果是深拷贝

则在内存里有两个对象的Name,但是不是指向源对象的Name的地址

也就是有两个内存不同位置的"张三"存在

实现深拷贝的三种方法是什么?

实现Cloneable接口并重写里面的clone()方法:

对象实现Cloneable接口并重写clone()方法,同时对象里面的引用字符类型也要实现Cloneable接口,重写clone()方法

java 复制代码
class MyClass implements Cloneable {
    private String field1;
    private NestedClass nestedObject;

    @Override
    protected Object clone() throws CloneNotSupportedException {
        MyClass cloned = (MyClass) super.clone();//深拷贝一层
        cloned.nestedObject = (NestedClass) nestedObject.clone(); // 深拷贝内部的引用对象
        return cloned;
    }
}

class NestedClass implements Cloneable {
    private int nestedField;

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

克隆时应该递归克隆,上述只克隆了一层,遇到引用类型 → 自动调用 clone (),不管深度多少层。

java 复制代码
class A implements Cloneable {
    B b;

    @Override
    public Object clone() {
        A a = (A) super.clone();
        a.b = (B) b.clone(); // 递归:B 也会 clone 它内部的对象
        return a;
    }
}

class B implements Cloneable {
    C c;

    @Override
    public Object clone() {
        B b = (B) super.clone();
        b.c = (C) c.clone(); // 继续递归
        return b;
    }
}

class C implements Cloneable {
    // ...
}

序列化和反序列化,实现Serializable接口

首先实现Serializable接口(表明可以序列化),然后把整个对象序列化,写入流,然后读出来生成新的对象

java 复制代码
import java.io.*;

class MyClass implements Serializable {
    private String field1;
    private NestedClass nestedObject;

    public MyClass deepCopy() {
        try {
            ByteArrayOutputStream bos = new ByteArrayOutputStream();//创建字节数组存放数据
            ObjectOutputStream oos = new ObjectOutputStream(bos);//创建对象输出流
            oos.writeObject(this);
            oos.flush();
            oos.close();

            ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());//从内存字节数组读取
            ObjectInputStream ois = new ObjectInputStream(bis);//创建输入流
            return (MyClass) ois.readObject();
        } catch (IOException | ClassNotFoundException e) {
            e.printStackTrace();
            return null;
        }
    }
}

class NestedClass implements Serializable {
    private int nestedField;
}

注意:try前半段的流程是,先创建好两个流,执行到第三行时,先写入当前对象,然后由ObjectOutputStream()把对象转化成字节(序列化),再传给ByteArrayOutputStream(),存入字节数组

手动递归克隆

与第一种方法类似,都是有多少个引用类型的字符串就写多少个递归函数,但不用实现接口和重写方法,适用于复杂度不高的情况

java 复制代码
class MyClass {
    private String field1;
    private NestedClass nestedObject;

    public MyClass deepCopy() {
        MyClass copy = new MyClass();
        copy.setField1(this.field1);
        copy.setNestedObject(this.nestedObject.deepCopy());
        return copy;
    }
}

class NestedClass {
    private int nestedField;

    public NestedClass deepCopy() {
        NestedClass copy = new NestedClass();
        copy.setNestedField(this.nestedField);
        return copy;
    }
}

方法一三实现操作都类似,但方法三不用重写方法与实现接口,再复杂度不高的时候选方法三,较高的时候选方法二

相关推荐
tsyjjOvO3 小时前
【SpringMVC 进阶】拦截器、文件上传、异常处理与 SSM 整合全解析
java·后端·spring
222you3 小时前
JUC读写锁和阻塞队列
java·开发语言·spring
太阳之神aboluo3 小时前
RabbitMQ
java·分布式·spring·rabbitmq·java-rabbitmq
qqty12173 小时前
windows配置永久路由
java
weixin_404157683 小时前
Java高级面试与工程实践问题集(七)
java·开发语言·面试
计算机学姐3 小时前
基于SpringBoot+Vue的智能民宿预定游玩系统【AI智能客服+数据可视化】
java·vue.js·spring boot·后端·mysql·spring·信息可视化
骇客野人3 小时前
JDK8和JDK8以后对jdk的优化,以及为什么如此优化
java·开发语言·windows
要努力点3 小时前
26考研——计算机考研复试——0854(2)
java·c语言·考研·算法·复试