对象克隆是指创建对象的精确副本。它创建当前对象类的新实例,并使用该对象相应字段的内容来初始化其所有字段。
在 Java 中,没有用于创建对象副本的运算符。与 C++ 不同,在 Java 中,如果我们使用赋值运算符,那么它将创建引用变量的副本而不是对象。这可以举个例子来解释。下面的程序演示了相同的内容。
// Java program to demonstrate that assignment operator
// only creates a new reference to same object
import java.io.*;
// A test class whose objects are cloned
class Test {
int x, y;
Test()
{
x = 10;
y = 20;
}
}
// Driver Class
class Main {
public static void main(String[] args)
{
Test ob1 = new Test();
System.out.println(ob1.x + " " + ob1.y);
// Creating a new reference variable ob2
// pointing to same address as ob1
Test ob2 = ob1;
// Any change made in ob2 will
// be reflected in ob1
ob2.x = 100;
System.out.println(ob1.x + " " + ob1.y);
System.out.println(ob2.x + " " + ob2.y);
}
}
输出
10 20
100 20
100 20
要制作其对象副本的类必须在其中或其父类之一中具有公共克隆方法。
每个实现 clone() 的类都应该调用 super.clone() 来获得克隆的对象引用。
该类还必须实现 java.lang.Cloneable 接口,我们要创建其对象克隆,否则当对该类的对象调用克隆方法时,它将抛出 CloneNotSupportedException。
句法:
受保护的对象克隆()抛出 CloneNotSupportedException
请注意——在下面的代码示例中,clone() 方法确实创建了一个具有不同 hashCode 值的全新对象,这意味着它位于单独的内存位置。但是由于Test对象c在Test2内部,原始类型实现了深拷贝,但是这个Test对象c仍然在t1和t2之间共享。为了克服这个问题,我们显式地为对象变量 c 做了一个深拷贝,这将在后面讨论。
// A Java program to demonstrate
// shallow copy using clone()
import java.util.ArrayList;
// An object reference of this class is
// contained by Test2
class Test {
int x, y;
}
// Contains a reference of Test and
// implements clone with shallow copy.
class Test2 implements Cloneable {
int a;
int b;
Test c = new Test();
public Object clone() throws CloneNotSupportedException
{
return super.clone();
}
}
// Driver class
public class Main {
public static void main(String args[])
throws CloneNotSupportedException
{
Test2 t1 = new Test2();
t1.a = 10;
t1.b = 20;
t1.c.x = 30;
t1.c.y = 40;
Test2 t2 = (Test2)t1.clone();
// Creating a copy of object t1
// and passing it to t2
t2.a = 100;
// Change in primitive type of t2 will
// not be reflected in t1 field
t2.c.x = 300;
// Change in object type field will be
// reflected in both t2 and t1(shallow copy)
System.out.println(t1.a + " " + t1.b + " " + t1.c.x + " " + t1.c.y);
System.out.println(t2.a + " " + t2.b + " " + t2.c.x
+ " " + t2.c.y);
}
}
输出
10 20 300 40
100 20 300 40
在上面的示例中,t1.clone 返回对象 t1 的浅表副本。要获得对象的深层副本,必须在获得副本后在克隆方法中进行某些修改。
浅拷贝是复制对象的方法,在克隆中默认遵循。在此方法中,旧对象 X 的字段被复制到新对象 Y。在复制对象类型字段时,引用被复制到 Y,即对象 Y 将指向与 X 所指出的相同位置。如果字段值是原始类型,它复制原始类型的值。
因此,对对象 X 或 Y 中的引用对象所做的任何更改都将反映在其他对象中。
浅拷贝便宜且制作简单。在上面的示例中,我们创建了对象的浅表副本。
如果我们想创建对象 X 的深层副本并将其放置在新对象 Y 中,则会创建任何引用对象字段的新副本,并将这些引用放置在对象 Y 中。这意味着在对象中引用的对象字段中所做的任何更改X 或 Y 将仅反映在该对象中,而不会反映在另一个对象中。在下面的示例中,我们创建了对象的深拷贝。
深拷贝复制所有字段并制作字段指向的动态分配内存的副本。当一个对象与其所引用的对象一起被复制时,就会发生深拷贝。
// A Java program to demonstrate
// deep copy using clone()
// An object reference of this
// class is contained by Test2
class Test {
int x, y;
}
// Contains a reference of Test and
// implements clone with deep copy.
class Test2 implements Cloneable {
int a, b;
Test c = new Test();
public Object clone() throws CloneNotSupportedException
{
// Assign the shallow copy to
// new reference variable t
Test2 t = (Test2)super.clone();
// Creating a deep copy for c
t.c = new Test();
t.c.x = c.x;
t.c.y = c.y;
// Create a new object for the field c
// and assign it to shallow copy obtained,
// to make it a deep copy
return t;
}
}
public class Main {
public static void main(String args[])
throws CloneNotSupportedException
{
Test2 t1 = new Test2();
t1.a = 10;
t1.b = 20;
t1.c.x = 30;
t1.c.y = 40;
Test2 t3 = (Test2)t1.clone();
t3.a = 100;
// Change in primitive type of t2 will
// not be reflected in t1 field
t3.c.x = 300;
// Change in object type field of t2 will
// not be reflected in t1(deep copy)
System.out.println(t1.a + " " + t1.b + " " + t1.c.x
+ " " + t1.c.y);
System.out.println(t3.a + " " + t3.b + " " + t3.c.x
+ " " + t3.c.y);
}
}
输出
10 20 30 40
100 20 300 40
在上面的示例中,我们可以看到已经为 Test 类分配了一个新对象来复制一个对象,该对象将返回给 clone 方法。因此,t3 将获得对象 t1 的深拷贝。因此,t3 对“c”对象字段所做的任何更改都不会反映在 t1 中。
如果我们使用赋值运算符将一个对象引用赋给另一个引用变量,那么它将指向旧对象的相同地址位置,并且不会创建该对象的新副本。因此,引用变量中的任何更改都将反映在原始对象中。
如果我们使用复制构造函数,那么我们必须显式复制所有数据,即我们必须显式地在构造函数中重新分配类的所有字段。但是在克隆方法中,创建新副本的工作是由方法本身完成的。所以为了避免额外的处理,我们使用对象克隆。
你适合学Java吗?4大专业测评方法
代码逻辑 吸收能力 技术学习能力 综合素质
先测评确定适合在学习