主页 > 开源代码  > 

深拷贝与浅拷贝

深拷贝与浅拷贝

在 Java 中,对象拷贝是常见的操作,主要分为浅拷贝和深拷贝两种方式,下面为你详细介绍它们的概念、实现方式以及区别。

1. 浅拷贝(Shallow Copy) 1.1 概念

浅拷贝创建一个新对象,新对象的属性值会复制原对象的属性值。对于基本数据类型,会直接复制其值;而对于引用数据类型,只是复制引用,即新对象和原对象的引用数据类型属性指向同一个内存地址。这意味着如果修改新对象中引用数据类型属性的内容,原对象中对应的属性内容也会被修改。

1.2 实现方式

在 Java 中,要实现浅拷贝,可以让类实现 Cloneable 接口并重写 clone() 方法。Cloneable 接口是一个标记接口,它本身不包含任何方法,只是告诉 Java 虚拟机该类可以被克隆。

示例代码:

class Address { String street; String city; public Address(String street, String city) { this.street = street; this.city = city; } } class Person implements Cloneable { String name; int age; Address address; public Person(String name, int age, Address address) { this.name = name; this.age = age; this.address = address; } @Override protected Object clone() throws CloneNotSupportedException { return super.clone(); } } public class ShallowCopyExample { public static void main(String[] args) throws CloneNotSupportedException { Address address = new Address("123 Main St", "New York"); Person person1 = new Person("John", 30, address); // 进行浅拷贝 Person person2 = (Person) person1.clone(); // 修改 person2 的基本数据类型属性 person2.age = 31; // 修改 person2 的引用数据类型属性 person2.address.street = "456 Elm St"; System.out.println("Person1 age: " + person1.age); System.out.println("Person1 address street: " + person1.address.street); System.out.println("Person2 age: " + person2.age); System.out.println("Person2 address street: " + person2.address.street); } }

输出结果:

Person1 age: 30 Person1 address street: 456 Elm St Person2 age: 31 Person2 address street: 456 Elm St

从输出结果可以看出,修改 person2 的基本数据类型属性 age 不会影响 person1 的 age,但修改 person2 的引用数据类型属性 address 的 street 会同时影响 person1 的 address 的 street,这就是浅拷贝的特点。

2. 深拷贝(Deep Copy) 2.1 概念

深拷贝同样创建一个新对象,并且会递归地复制原对象的所有属性,包括基本数据类型和引用数据类型。对于引用数据类型,会创建一个新的对象并复制其内容,而不是简单地复制引用。因此,新对象和原对象的引用数据类型属性指向不同的内存地址,修改新对象中引用数据类型属性的内容不会影响原对象。

2.2 实现方式

实现深拷贝有多种方式,常见的是手动实现和使用序列化与反序列化。

手动实现

手动实现深拷贝需要在 clone() 方法中递归地复制引用数据类型的属性。

示例代码:

class Address implements Cloneable { String street; String city; public Address(String street, String city) { this.street = street; this.city = city; } @Override protected Object clone() throws CloneNotSupportedException { return super.clone(); } } class Person implements Cloneable { String name; int age; Address address; public Person(String name, int age, Address address) { this.name = name; this.age = age; this.address = address; } @Override protected Object clone() throws CloneNotSupportedException { Person clonedPerson = (Person) super.clone(); // 手动复制引用数据类型属性 clonedPerson.address = (Address) address.clone(); return clonedPerson; } } public class DeepCopyExample { public static void main(String[] args) throws CloneNotSupportedException { Address address = new Address("123 Main St", "New York"); Person person1 = new Person("John", 30, address); // 进行深拷贝 Person person2 = (Person) person1.clone(); // 修改 person2 的基本数据类型属性 person2.age = 31; // 修改 person2 的引用数据类型属性 person2.address.street = "456 Elm St"; System.out.println("Person1 age: " + person1.age); System.out.println("Person1 address street: " + person1.address.street); System.out.println("Person2 age: " + person2.age); System.out.println("Person2 address street: " + person2.address.street); } }

输出结果:

Person1 age: 30 Person1 address street: 123 Main St Person2 age: 31 Person2 address street: 456 Elm St

从输出结果可以看出,修改 person2 的基本数据类型属性 age 和引用数据类型属性 address 的 street 都不会影响 person1,这就是深拷贝的特点。

使用序列化与反序列化

使用 Java 的序列化和反序列化机制也可以实现深拷贝。需要让类实现 Serializable 接口,然后通过 ObjectOutputStream 和 ObjectInputStream 进行序列化和反序列化操作。

示例代码:

import java.io.*; class Address implements Serializable { String street; String city; public Address(String street, String city) { this.street = street; this.city = city; } } class Person implements Serializable { String name; int age; Address address; public Person(String name, int age, Address address) { this.name = name; this.age = age; this.address = address; } public Person deepCopy() throws IOException, ClassNotFoundException { ByteArrayOutputStream bos = new ByteArrayOutputStream(); ObjectOutputStream oos = new ObjectOutputStream(bos); oos.writeObject(this); ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray()); ObjectInputStream ois = new ObjectInputStream(bis); return (Person) ois.readObject(); } } public class DeepCopyWithSerializationExample { public static void main(String[] args) throws IOException, ClassNotFoundException { Address address = new Address("123 Main St", "New York"); Person person1 = new Person("John", 30, address); // 进行深拷贝 Person person2 = person1.deepCopy(); // 修改 person2 的基本数据类型属性 person2.age = 31; // 修改 person2 的引用数据类型属性 person2.address.street = "456 Elm St"; System.out.println("Person1 age: " + person1.age); System.out.println("Person1 address street: " + person1.address.street); System.out.println("Person2 age: " + person2.age); System.out.println("Person2 address street: " + person2.address.street); } }

这种方式的优点是代码简洁,不需要手动处理每个引用数据类型的复制;缺点是需要类实现 Serializable 接口,并且序列化和反序列化操作可能会影响性能。

3. 浅拷贝与深拷贝的区别 基本数据类型:浅拷贝和深拷贝都会复制基本数据类型的值,修改新对象的基本数据类型属性不会影响原对象。引用数据类型:浅拷贝只复制引用,新对象和原对象的引用数据类型属性指向同一个内存地址,修改新对象的引用数据类型属性会影响原对象;深拷贝会创建新的对象并复制其内容,新对象和原对象的引用数据类型属性指向不同的内存地址,修改新对象的引用数据类型属性不会影响原对象。

综上所述,在选择使用浅拷贝还是深拷贝时,需要根据具体的业务需求来决定。如果不希望修改新对象影响原对象,应该使用深拷贝;如果只关心对象的基本数据类型属性,或者可以接受引用数据类型属性的共享,那么浅拷贝就足够了。

标签:

深拷贝与浅拷贝由讯客互联开源代码栏目发布,感谢您对讯客互联的认可,以及对我们原创作品以及文章的青睐,非常欢迎各位朋友分享到个人网站或者朋友圈,但转载请说明文章出处“深拷贝与浅拷贝