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

目录

深拷贝和浅拷贝的区别?

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

实现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;
    }
}

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

相关推荐
土狗TuGou10 分钟前
SQL内功笔记 · 第8篇:事务的四大特性与隔离级别
数据库·笔记·后端·sql·mysql·oracle
ZengLiangYi19 分钟前
React Query + REST API 最佳实践
javascript·后端·react.js
星浩AI24 分钟前
项目实战:合同智能审批 · LangGraph + HITL 人机协同方案 [有源码]
后端·langchain·agent
JavaGuide26 分钟前
Codex 接入第三方模型 DeepSeek、GLM、Kimi 教程:CC-Switch 和 Codex++ 两种方案对比
后端·ai编程
ZengLiangYi28 分钟前
Fastify 加 Electron:把 Web 服务嵌进桌面应用
前端·javascript·后端
在繁华处1 小时前
Java从零到熟练(九):并发编程基础
java·开发语言
木头程序员1 小时前
SSM框架学习笔记
java·开发语言·mysql·spring·maven
李白你好1 小时前
页面资产梳理 · 技术指纹识别 · Spring 端点探测
java·后端·spring
用户1753721240331 小时前
02《面向对象设计原则:SOLID原则实战解析》
后端
一起逃去看海吧1 小时前
dify-03
java·linux·开发语言