Java的深浅拷贝认识

简介: 该内容是关于Java中深拷贝和浅拷贝的解释。浅拷贝创建新对象,但属性(包括引用类型)仍指向原对象属性,导致修改时互相影响。深拷贝则为所有引用类型创建副本,确保对象独立。深拷贝适用于写入操作和需要独立副本的场景。分辨深浅拷贝主要看赋值和方法参数传递,以及是否有创建新实例的逻辑。

目录


在Java中,深拷贝和浅拷贝是对象复制的两种方式,主要区别在于对对象内部的引用类型的处理上。

浅拷贝

定义:

浅拷贝是指创建一个新的对象,但这个新对象的属性(包括引用类型的属性)仍然指向原来对象的属性。换言之,如果原对象中的属性是一个引用类型,那么浅拷贝只会复制这个引用的地址,新旧对象会共享同一块内存区域。因此,修改其中一个对象的引用类型属性时,另一个对象的相同属性也会受到影响。

优点:

  • 省资源。

适合场景:

  • 只读取,不写入的场景;
  • 如果对象结构简单,或者希望节省资源,浅拷贝更合适;

code:

class Person implements Cloneable {
    String name;
    Address address; // 假设Address是一个类
    public Person clone() {
        try {
            return (Person) super.clone();
        } catch (CloneNotSupportedException e) {
            throw new AssertionError(); // 无法克隆的情况不应该发生
        }
    }
}

Object祖先类默认实现了浅拷贝,在不重写的情况下,使用构造函数或copy()等自定义方法, 都是直接赋值引用类型字段,都是属于浅拷贝。

深拷贝

定义:

深拷贝不仅会创建一个新对象,还会递归地为所有引用类型的属性创建副本,确保源对象和拷贝对象之间完全独立,修改一个对象不会影响到另一个对象。

优点:

  • 没有优点,只能说更适合写入操作;

适合场景:

  • 需要写入操作的场景;
  • 需要完全独立的副本,避免修改时的相互影响;

重写cloe方法实现深拷贝示例:

// 对于每个引用类型的属性,手动调用其拷贝方法(如果该类也支持深拷贝的话)来创建新的实例。
class Person implements Cloneable {
    String name;
    Address address;
    public Person deepClone() {
        Person clone = new Person();
        clone.name = this.name;
        clone.address = this.address.clone(); // 假设Address类也实现了深拷贝
        return clone;
    }
}

注意:深拷贝需要注意嵌套的引用类型是否也实现了重写问题,不然嵌套类还是对类型的引用的浅拷贝。

背景解释:

引用数据类型指的是类、接口、数组、String等数据类型,与之相对的是基本数据类型,如int、double、boolean等。

实现深拷贝的方法:

  • 重写clone()方法来实现深拷贝
  • 使用第三方库如Apache Commons Lang等
  • 手动实现深拷贝逻辑
  • 序列化方式实现深拷贝
  • ...

分辨代码里的深浅拷贝

看赋值:

  • 当一个对象实例被赋值给另一个变量时(例如 Object obj2 = obj1;),无论是在方法参数传递还是普通变量赋值中,如果不进行特殊处理,这总是浅拷贝。两个变量实际上指向内存中的同一个对象。

看方法参数传递:

  • 在Java中,方法参数传递本质上是值传递。当对象作为参数传递给方法时,传递的是对象引用的副本(即指针的拷贝),而不是对象本身的新拷贝。因此,方法内对对象内容的修改会影响原始对象,这是浅拷贝行为。但如果方法内部创建了对象的新实例并返回,则可以实现深拷贝。
  • 方法调用时,检查是否仅传递了引用还是进行了对象内容的实际复制。
  • 查看代码中是否有显式地为引用类型字段创建新实例的逻辑,这是深拷贝的关键标志。
相关文章
|
9月前
|
Java
【Java】数组中的拷贝方法与初步理解深浅拷贝
【Java】数组中的拷贝方法与初步理解深浅拷贝
56 0
|
Java
Java中的深浅拷贝问题,你清楚吗?
Java中的深浅拷贝问题,你清楚吗?
122 0
Java中的深浅拷贝问题,你清楚吗?
|
存储 JSON 缓存
5张图搞懂Java深浅拷贝
在开发、刷题、面试中,我们可能会遇到将一个对象的属性赋值到另一个对象的情况,这种情况就叫做拷贝。拷贝与Java内存结构息息相关,搞懂Java深浅拷贝是很必要的!
243 0
5张图搞懂Java深浅拷贝
|
存储 Java
java提高(15)---java深浅拷贝
java提高(15)---java深浅拷贝 一、前言 为什么会有深浅拷贝这个概念? 我觉得主要跟JVM内存分配有关,对于基本数据类型,只存在栈内存,所以它的拷贝不存在深浅拷贝这个概念。而对于对象而言,一个对象的创建会在内存中分配两块空间,一个在栈内存存对象的引用指针,一个在堆内存存放对象。
9369 0
|
2天前
|
Java 程序员 开发者
Java社招面试题:一个线程运行时发生异常会怎样?
大家好,我是小米。今天分享一个经典的 Java 面试题:线程运行时发生异常,程序会怎样处理?此问题考察 Java 线程和异常处理机制的理解。线程发生异常,默认会导致线程终止,但可以通过 try-catch 捕获并处理,避免影响其他线程。未捕获的异常可通过 Thread.UncaughtExceptionHandler 处理。线程池中的异常会被自动处理,不影响任务执行。希望这篇文章能帮助你深入理解 Java 线程异常处理机制,为面试做好准备。如果你觉得有帮助,欢迎收藏、转发!
35 14
|
5天前
|
安全 Java 程序员
Java 面试必问!线程构造方法和静态块的执行线程到底是谁?
大家好,我是小米。今天聊聊Java多线程面试题:线程类的构造方法和静态块是由哪个线程调用的?构造方法由创建线程实例的主线程调用,静态块在类加载时由主线程调用。理解这些细节有助于掌握Java多线程机制。下期再见! 简介: 本文通过一个常见的Java多线程面试题,详细讲解了线程类的构造方法和静态块是由哪个线程调用的。构造方法由创建线程实例的主线程调用,静态块在类加载时由主线程调用。理解这些细节对掌握Java多线程编程至关重要。
34 13
|
6天前
|
安全 Java 开发者
【JAVA】封装多线程原理
Java 中的多线程封装旨在简化使用、提高安全性和增强可维护性。通过抽象和隐藏底层细节,提供简洁接口。常见封装方式包括基于 Runnable 和 Callable 接口的任务封装,以及线程池的封装。Runnable 适用于无返回值任务,Callable 支持有返回值任务。线程池(如 ExecutorService)则用于管理和复用线程,减少性能开销。示例代码展示了如何实现这些封装,使多线程编程更加高效和安全。
|
1月前
|
监控 Java
java异步判断线程池所有任务是否执行完
通过上述步骤,您可以在Java中实现异步判断线程池所有任务是否执行完毕。这种方法使用了 `CompletionService`来监控任务的完成情况,并通过一个独立线程异步检查所有任务的执行状态。这种设计不仅简洁高效,还能确保在大量任务处理时程序的稳定性和可维护性。希望本文能为您的开发工作提供实用的指导和帮助。
109 17
|
2月前
|
Java
Java—多线程实现生产消费者
本文介绍了多线程实现生产消费者模式的三个版本。Version1包含四个类:`Producer`(生产者)、`Consumer`(消费者)、`Resource`(公共资源)和`TestMain`(测试类)。通过`synchronized`和`wait/notify`机制控制线程同步,但存在多个生产者或消费者时可能出现多次生产和消费的问题。 Version2将`if`改为`while`,解决了多次生产和消费的问题,但仍可能因`notify()`随机唤醒线程而导致死锁。因此,引入了`notifyAll()`来唤醒所有等待线程,但这会带来性能问题。
Java—多线程实现生产消费者