深拷贝和浅拷贝

32 阅读2分钟

深拷贝和浅拷贝

拷贝对象,有两种方式:浅拷贝和深拷贝

  • 浅拷贝:在堆上创建一个新的对象(区别于引用拷贝的一点),如果原对象内部的属性是引用类型的话,浅拷贝会直接复制内部对象的引用地址,也就是说拷贝对象和原对象共用同一个内部对象。原对象内部的属性是基础数据类型,可以直接复制,原对象修改后,新对象也会随之修改对应的基础数据类型的值。
  • 深拷贝 :完全复制整个对象,包括该对象所包含的引用类型。

浅拷贝

浅拷贝的示例代码如下,实现了 Cloneable 接口,并重写了 clone() 方法。 clone() 方法的实现很简单,直接调用的是父类 Object 的 clone() 方法。

public class Address implements Cloneable{
    private String name;
    // 省略构造函数、Getter&Setter方法
    @Override
    public Address clone() {
        try {
            return (Address) super.clone();
        } catch (CloneNotSupportedException e) {
            throw new AssertionError();
        }
    }
}

public class Person implements Cloneable {
    private Address address;
    // 省略构造函数、Getter&Setter方法
    @Override
    public Person clone() {
        try {
            Person person = (Person) super.clone();
            return person;
        } catch (CloneNotSupportedException e) {
            throw new AssertionError();
        }
    }
}

测试 :

Person person1 = new Person(new Address("北京"));
Person person1Copy = person1.clone();
// true
System.out.println(person1.getAddress() == person1Copy.getAddress());

person1的成员变量有引用数据类型Address类的成员变量address,因此浅拷贝只复制该对象的引用地址。从输出结构就可以看出, person1 的克隆对象和 person1 使用的仍然是同一个 Address 对象。

深拷贝 这里我们简单对 Person 类的 clone() 方法进行修改,连带着要把 Person 对象内部的 Address 对象一起复制。

@Override
public Person clone() {
    try {
        Person person = (Person) super.clone();
        person.setAddress(person.getAddress().clone());
        return person;
    } catch (CloneNotSupportedException e) {
        throw new AssertionError();
    }
}

测试

Person person1 = new Person(new Address("北京"));
Person person1Copy = person1.clone();
System.out.println(person.getAddress());        // copy.Address@4554617c
System.out.println(personCopy.getAddress());    // copy.Address@74a14482
// false
System.out.println(person1.getAddress() == person1Copy.getAddress());

很明显,person1 的克隆对象和 person1 包含的 Address 对象已经是不同的了

那什么是引用拷贝呢?  简单来说,引用拷贝就是两个不同的引用指向同一个对象。

浅拷贝、深拷贝、引用拷贝图示 image.png