一文搞清深拷贝和浅拷贝

一文搞清深拷贝和浅拷贝


对象拷贝在我们日常写代码的时候基本上是刚性需求,经常遇到,只不过很多人天天忙于写业务,忽视了一些细节问题和理解,有时候这方面一旦出了问题,就不太容易排查了。

在这里插入图片描述

值类型 vs 引用类型

这两个概念的准确区分,对于深、浅拷贝问题的理解非常重要。

正如Java圣经《Java编程思想》第二章的标题所言,在Java中一切都可以视为对象!

所以来到Java的世界,我们要习惯用引用去操作对象。在Java中,像数组、类Class、枚举EnumInteger包装类等等,就是典型的引用类型,所以操作时一般来说采用的也是引用传递的方式;

但是Java的语言级基础数据类型,诸如int这些基本类型,操作时一般采取的则是值传递的方式,所以有时候也称它为值类型。

为了便于下文的讲述和举例,我们这里先定义两个类:StudentMajor,分别表示「学生」以及「所学的专业」,二者是包含关系:

// 学生的所学专业
public class Major {
    private String majorName; // 专业名称
    private long majorId;     // 专业代号
    
    // ... 其他省略 ...
}
// 学生
public class Student {
    private String name;  // 姓名
    private int age;      // 年龄
    private Major major;  // 所学专业
    
    // ... 其他省略 ...
}

在这里插入图片描述

赋值 vs 浅拷贝 vs 深拷贝

对象赋值

赋值是日常编程过程中最常见的操作,最简单的比如:

Student codeSheep = new Student();
Student codePig = codeSheep;

严格来说,这种不能算是对象拷贝,因为拷贝的仅仅只是引用关系,并没有生成新的实际对象:
在这里插入图片描述

浅拷贝

浅拷贝属于对象克隆方式的一种,重要的特性体现在这个「浅」字上。

比如我们试图通过studen1实例,拷贝得到student2,如果是浅拷贝这种方式,大致模型可以示意成如下所示的样子:
在这里插入图片描述

很明显,值类型的字段会复制一份,而引用类型的字段拷贝的仅仅是引用地址,而该引用地址指向的实际对象空间其实只有一份。

一图胜前言,我想上面这个图已经表现得很清楚了。

深拷贝

深拷贝相较于上面所示的浅拷贝,除了值类型字段会复制一份,引用类型字段所指向的对象,会在内存中也创建一个副本,就像这个样子:
在这里插入图片描述

原理很清楚明了,下面来看看具体的代码实现吧。

浅拷贝代码实现

还以上文的例子来讲,我想通过student1拷贝得到student2,浅拷贝的典型实现方式是:让被复制对象的类实现Cloneable接口,并重写clone()方法即可。

以上面的Student类拷贝为例:

public class Student implements Cloneable {

    private String name;  // 姓名
    private int age;      // 年龄
    private Major major;  // 所学专业

    @Override
    public Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
    
    // … 其他省略 …

}

然后我们写个测试代码,一试便知:

public class Test {

    public static void main(String[] args) throws CloneNotSupportedException {

        Major m = new Major("计算机科学与技术",666666);
        Student student1 = new Student( "CodeSheep", 18, m );
        
        // 由 student1 拷贝得到 student2
        Student student2 = (Student) student1.clone();

        System.out.println( student1 == student2 );
        System.out.println( student1 );
        System.out.println( student2 );
        System.out.println( "\n" );

        // 修改student1的值类型字段
        student1.setAge( 35 );
        
        // 修改student1的引用类型字段
        m.setMajorName( "电子信息工程" );
        m.setMajorId( 888888 );

        System.out.println( student1 );
        System.out.println( student2 );

    }
}

运行得到如下结果:
在这里插入图片描述
从结果可以看出:

  • student1==student2打印false,说明clone()方法的确克隆出了一个新对象;
  • 修改值类型字段并不影响克隆出来的新对象,符合预期;
  • 而修改了student1内部的引用对象,克隆对象student2也受到了波及,说明内部还是关联在一起的

深拷贝代码实现

深度遍历式拷贝

虽然clone()方法可以完成对象的拷贝工作,但是注意:clone()方法默认是浅拷贝行为,就像上面的例子一样。若想实现深拷贝需覆写clone()方法实现引用对象的深度遍历式拷贝,进行地毯式搜索。

所以对于上面的例子,如果想实现深拷贝,首先需要对更深一层次的引用类Major做改造,让其也实现Cloneable接口并重写clone()方法:

public class Major implements Cloneable {

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
    
    // ... 其他省略 ...
}

其次我们还需要在顶层的调用类中重写clone()方法,来调用引用类型字段的clone()方法实现深度拷贝,对应到本文那就是Student类:

public class Student implements Cloneable {

    @Override
    public Object clone() throws CloneNotSupportedException {
        Student student = (Student) super.clone();
        student.major = (Major) major.clone(); // 重要!!!
        return student;
    }
    
    // ... 其他省略 ...
}

这时候上面的测试用例不变,运行可得结果:
在这里插入图片描述
很明显,这时候student1student2两个对象就完全独立了,不受互相的干扰。

利用反序列化实现深拷贝

记得在前文《序列化/反序列化,我忍你很久了》中就已经详细梳理和总结了「序列化和反序列化」这个知识点了。

利用反序列化技术,我们也可以从一个对象深拷贝出另一个复制对象,而且这货在解决多层套娃式的深拷贝问题时效果出奇的好。

所以我们这里改造一下Student类,让其clone()方法通过序列化和反序列化的方式来生成一个原对象的深拷贝副本:

public class Student implements Serializable {

    private String name;  // 姓名
    private int age;      // 年龄
    private Major major;  // 所学专业

    public Student clone() {
        try {
            // 将对象本身序列化到字节流
            ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
            ObjectOutputStream objectOutputStream =
                    new ObjectOutputStream( byteArrayOutputStream );
            objectOutputStream.writeObject( this );

            // 再将字节流通过反序列化方式得到对象副本
            ObjectInputStream objectInputStream =
                    new ObjectInputStream( new ByteArrayInputStream( byteArrayOutputStream.toByteArray() ) );
            return (Student) objectInputStream.readObject();

        } catch (IOException e) {
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }

        return null;
    }
    
    // ... 其他省略 ...
}

当然这种情况下要求被引用的子类(比如这里的Major类)也必须是可以序列化的,即实现了Serializable接口:

public class Major implements Serializable {
  
  // ... 其他省略 ...
    
}

这时候测试用例完全不变,直接运行,也可以得到如下结果:
在这里插入图片描述
很明显,这时候student1student2两个对象也是完全独立的,不受互相的干扰,深拷贝完成。

<think>首先,用户的问题是:“我想了解深拷贝浅拷贝的使用场景 请问深拷贝浅拷贝的区别及适用场景”。这基于系统级指令,我需要用中文回答,结构清晰,逐步解决问题。 关键点: - **区别**:深拷贝浅拷贝的核心区别在于如何处理对象中的引用类型属性。 - **浅拷贝**:只复制对象的第一层属性。如果属性是基本类型(如数字、字符串),则复制值;如果属性是引用类型(如对象、数组),则复制引用(内存地址),所以新对象原对象共享这些引用类型属性。 - **深拷贝**:递归复制对象的所有层级属性。创建一个完全独立的新对象,包括所有嵌套对象,没有任何共享引用。 - **适用场景**:根据引用来分析。 - 引用[1]:深拷贝浅拷贝在不同场景下各有优劣。理解原理区别,帮助选择合适的拷贝方式。 - 引用[2]:解释深拷贝浅拷贝的区别。 - 引用[3]:深拷贝更复杂耗时,但提供更高安全性独立性;浅拷贝更快。 - 引用[4]:浅拷贝适用于简单复制操作;深拷贝适用于复杂嵌套结构。 - 引用[5]:表格总结了特性:浅拷贝快,适用于简单对象;深拷贝慢,适用于复杂对象。 回答结构: 1. **引言**:简要介绍深拷贝浅拷贝的概念。 2. **区别**:详细解释核心区别。 3. **适用场景**:分别列出浅拷贝深拷贝的适用场景。 4. **示例**:提供简单代码示例(如Python或JavaScript,因为引用中提到这些)。 5. **结论**:总结如何选择。 6. **相关问题**:生成相关问题,如系统级指令要求。 系统级指令要求: - 行内数学表达式用$...$,但这里可能不涉及数学表达式。 - 独立公式用$$...$$,但同样不相关。 - LaTeX语法正确:确保任何数学格式正确。 - 使用中文回答。 - 生成相关问题:在末尾添加“§§相关问题§§”部分。 - 回答中引用的段落末尾自然地添加引用标识,如[^1]。 引用来源: - 引用[1]:一文吃透深拷贝浅拷贝:原理、区别及应用场景。 - 引用[2]:解释Java中的深拷贝浅拷贝有什么区别? - 引用[3]:深拷贝浅拷贝的区别及实现方式。 - 引用[4]:深拷贝浅拷贝的区别及应用场景详解。 - 引用[5]:特性对比表格。 在回答中,我需要自然地引用这些,例如在解释区别或场景时。 完整回答大纲: 1. **介绍**:定义深拷贝浅拷贝。 2. **区别**: - 浅拷贝:只复制第一层,共享嵌套对象。 - 深拷贝:完全复制所有层级,独立对象。 - 引用[5]的表格作为参考。 3. **适用场景**: - 浅拷贝场景:当对象简单,没有嵌套引用类型时;性能要求高时。 - 深拷贝场景:当对象复杂,有嵌套对象;需要完全独立副本时。 - 引用[3][4]来支持。 4. **示例**:提供一个代码示例。 5. **总结**:如何选择。 6. **相关问题**:生成2-3个相关问题。 代码示例:使用JavaScript或Python,因为常见。 - JavaScript:浅拷贝如Object.assign()或展开运算符;深拷贝如JSON.parse(JSON.stringify())或递归函数。 - Python:浅拷贝如copy.copy();深拷贝如copy.deepcopy()。 最后,添加相关问题。</think>### 深拷贝浅拷贝的区别及适用场景 在编程中,对象复制是常见操作,深拷贝浅拷贝是两种核心策略,它们的主要区别在于**如何处理对象中的嵌套引用类型属性(如对象、数组)**。理解它们的区别适用场景,能帮助开发者避免数据共享导致的意外错误,并优化性能。下面我将逐步解释核心概念、区别、适用场景,并提供示例。 #### 1. **基本概念** - **浅拷贝(Shallow Copy)**:只复制对象的第一层属性。如果属性是基本类型(如数字、字符串),则复制值;如果属性是引用类型(如对象、数组),则复制引用(内存地址),导致新对象原对象共享这些嵌套属性。修改拷贝后的嵌套属性会影响原对象[^2]。 - **深拷贝(Deep Copy)**:递归复制对象的所有层级属性,包括嵌套引用类型。创建一个完全独立的新对象,没有任何共享引用。修改拷贝后的对象不会影响原对象[^3]。 #### 2. **核心区别** 深拷贝浅拷贝的关键区别体现在以下方面: - **复制层级**: - 浅拷贝仅复制第一层属性(如对象的顶层键值对)。 - 深拷贝递归复制所有层级(包括嵌套对象、数组等)。 - **内存引用**: - 浅拷贝中,嵌套引用类型共享原对象的内存地址。 - 深拷贝中,所有属性都是独立的,无内存共享。 - **修改影响**: - 浅拷贝后,修改嵌套属性会影响原对象(例如,改变拷贝数组的元素会同步改变原数组)。 - 深拷贝后,修改任何属性都不会影响原对象。 - **性能**: - 浅拷贝速度快,因为它只处理第一层(时间复杂度通常为$O(n)$,其中$n$是第一层属性数量)。 - 深拷贝速度慢,因为它需要递归遍历所有层级(时间复杂度可能为$O(m)$,其中$m$是对象总属性数,包括嵌套层)[^5]。 - **适用对象类型**: - 浅拷贝适合简单对象(无嵌套引用类型)。 - 深拷贝适合复杂对象(有嵌套对象、数组等)[^4]。 总结区别表格: | 特性 | 浅拷贝 | 深拷贝 | |------|--------|--------| | **复制层级** | 仅第一层属性 | 所有层级属性(递归) | | **内存引用** | 嵌套对象共享原对象地址 | 嵌套对象完全独立 | | **修改影响** | 修改拷贝对象可能影响原对象 | 修改拷贝对象不影响原对象 | | **性能** | 快($O(n)$) | 慢($O(m)$) | | **适用对象** | 简单对象(如无嵌套的键值对) | 复杂对象(如多层嵌套结构)[^5] | #### 3. **适用场景** 根据对象结构性能需求,选择合适的方式: - **浅拷贝适用场景**: - **对象结构简单**:当对象只包含基本类型属性(如数字、字符串),没有嵌套引用类型时。例如,配置对象或数据传输对象(DTO)。 - **性能敏感场景**:在需要高效复制的场景,如高频数据处理或实时系统,浅拷贝能减少开销[^3]。 - **临时共享数据**:当多个部分需要访问同一数据,且允许共享修改时(如状态管理中的部分更新)。 示例场景:复制一个用户配置对象(只有`name``age`属性),使用浅拷贝即可避免不必要的递归开销[^4]。 - **深拷贝适用场景**: - **对象结构复杂**:当对象包含嵌套引用类型(如对象中的数组、子对象)时,深拷贝确保完全独立,避免意外修改。 - **数据隔离需求**:在需要安全隔离的场景,如状态管理(Redux)、缓存系统或并发编程,确保拷贝对象不受原对象影响。 - **持久化或序列化**:当对象需要存储或传输(如JSON转换)时,深拷贝防止引用丢失或污染[^1]。 示例场景:复制一个订单对象(包含嵌套的`items`数组`customer`对象),使用深拷贝保证订单数据修改不会影响原对象[^3]。 #### 4. **代码示例(JavaScript)** 以下示例展示浅拷贝深拷贝的实现及效果: ```javascript // 原对象(包含嵌套属性) const original = { name: "Alice", details: { age: 30, hobbies: ["reading", "coding"] } }; // 浅拷贝示例(使用展开运算符) const shallowCopy = { ...original }; shallowCopy.details.age = 31; // 修改嵌套属性 console.log(original.details.age); // 输出: 31(原对象被影响) // 深拷贝示例(使用JSON方法,注意:此方法不处理函数或循环引用) const deepCopy = JSON.parse(JSON.stringify(original)); deepCopy.details.hobbies.push("gaming"); // 修改嵌套属性 console.log(original.details.hobbies); // 输出: ["reading", "coding"](原对象不受影响) ``` - **解释**:浅拷贝中,修改`shallowCopy.details.age`会改变`original.details.age`,因为它们共享引用;深拷贝中,修改`deepCopy`不会影响原对象[^4]。 #### 5. **如何选择** - **优先浅拷贝**:如果对象简单且无嵌套引用类型,或性能是关键因素(如大数据处理)。 - **优先深拷贝**:如果对象复杂、需要数据隔离或涉及并发操作。 - **权衡**:深拷贝虽安全,但可能带来性能开销;浅拷贝高效,但需注意共享风险。在实际开发中,结合工具如Lodash的`_.cloneDeep()`(深拷贝)或语言内置方法(如Python的`copy.deepcopy()`)来简化实现[^1][^3]。 理解这些原理,能帮助您在编程中避免常见错误(如意外数据修改),并优化应用性能。如果您有具体语言或场景的疑问,欢迎提供更多细节!
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值