在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中,方法参数传递本质上是值传递。当对象作为参数传递给方法时,传递的是对象引用的副本(即指针的拷贝),而不是对象本身的新拷贝。因此,方法内对对象内容的修改会影响原始对象,这是浅拷贝行为。但如果方法内部创建了对象的新实例并返回,则可以实现深拷贝。

  • 方法调用时,检查是否仅传递了引用还是进行了对象内容的实际复制。

  • 查看代码中是否有显式地为引用类型字段创建新实例的逻辑,这是深拷贝的关键标志。

注意:基本数据类型与引用类型有区别,基本数据类型的赋值总是按值传递,不会涉及深浅拷贝问题。

05-24 07:54