Java 中的 clone() 方法\
java programming java8object oriented programming更新于 2024/10/22 20:07:00
Java 提供了一个赋值运算符来复制值,但没有提供运算符来复制对象。Object 类有一个 clone 方法,可用于复制对象的值而没有任何副作用。赋值运算符有一个副作用,即当将引用分配给另一个引用时,不会创建新对象,并且两个引用都指向同一个对象。这意味着如果我们更改一个对象中的值,那么相同的值也会反映在另一个对象中。clone() 方法处理了这个问题。请参阅以下示例。
示例
public class Tester { public static void main(String[] args) throws CloneNotSupportedException { //场景 1:使用赋值运算符复制对象 A a1 = new A(); a1.a = 1; a1.b = 2; //打印 a1 对象 System.out.println("a1: [" + a1.a + ", " + a1.b + "]"); //赋值运算符复制引用 A a2 = a1; //a2 和 a1 现在指向同一个对象 //修改 a2,更改将反映在 a1 中 a2.a = 3; System.out.println("a1: [" + a1.a + ", " + a1.b + "]"); System.out.println("a2: [" + a2.a + ", " + a2.b + "]"); //场景2:使用克隆,我们可以避免上述问题 B b1 = new B(); b1.a = 1; b1.b = 2; //打印 b1 对象 System.out.println("b1: [" + b1.a + ", " + b1.b + "]"); //克隆方法复制对象 B b2 = b1.clone(); //b2 和 b1 现在指向不同的对象 //修改 b2,更改不会反映在 b1 中 b2.a = 3; System.out.println("b1: [" + b1.a + ", " + b1.b + "]"); System.out.println("b2: [" + b2.a + ", " + b2.b + "]"); } } class A { public int a; public int b; } class B implements Cloneable { public int a; public int b; public B clone() throws CloneNotSupportedException { B b = (B)super.clone(); return b; } }
输出
a1: [1, 2] a1: [3, 2] a2: [3, 2] b1: [1, 2] b1: [1, 2] b2: [3, 2]
要记住的要点
我们可以使用赋值运算符以及 clone() 方法复制对象。
赋值运算符有副作用,因为它只复制引用,底层对象保持不变。
对于原始实例变量,clone() 方法没有副作用,因为在克隆过程中会创建一个新对象。
如果未正确实现 clone() 方法,对于作为实例变量的对象,也会产生副作用,因为克隆的对象具有引用的副本。这称为浅拷贝。
可以重写 clone() 方法来防止浅拷贝,可以单独创建实例变量对象并更新其属性。这称为深拷贝。