1.说明
protected Object clone() throws CloneNotSupportedException
创建并返回此对象的一个副本。“副本”的准确含义可能依赖于对象的类。一般来说,对于任何对象 x,如果表达式:
x.clone() != x
是正确的,则表达式:
x.clone().getClass() == x.getClass()
将为 true,但这些不是绝对条件。一般情况下是:
x.clone().equals(x)
将为 true,但这不是绝对条件。
按照惯例,返回的对象应该通过调用 super.clone 获得。如果一个类及其所有的超类(Object 除外)都遵守此约定,则 x.clone().getClass() == x.getClass()。
按照惯例,此方法返回的对象应该独立于该对象(正被克隆的对象)。要获得此独立性,在 super.clone 返回对象之前,有必要对该对象的一个或多个字段进行修改。这通常意味着要复制包含正在被克隆对象的内部“深层结构”的所有可变对象,并使用对副本的引用替换对这些对象的引用。如果一个类只包含基本字段或对不变对象的引用,那么通常不需要修改 super.clone 返回的对象中的字段。
Object 类的 clone 方法执行特定的克隆操作。首先,如果此对象的类不能实现接口 Cloneable,则会抛出 CloneNotSupportedException。注意:所有的数组都被视为实现接口 Cloneable。否则,此方法会创建此对象的类的一个新实例,并像通过分配那样,严格使用此对象相应字段的内容初始化该对象的所有字段;这些字段的内容没有被自我克隆。所以,此方法执行的是该对象的“浅表复制”,而不“深层复制”操作。
Object 类本身不实现接口 Cloneable,所以在类为 Object 的对象上调用 clone 方法将会导致在运行时抛出异常。
2.实现
1)implements Cloneable
2)覆盖clone方法
public Object clone() throws CloneNotSupportedException {
return super.clone();
}
3.实质
浅复制:将对象在栈中的数据复制一份。
结果:对对象的基本类型和String(特殊的引用对象)属性变更时不影响clone对象,而对引用属性变更时会影响clone对象,所以想要实现深复制就要自己重写clone方法。
4.实例
package com.siyuan.test.clone;
class A implements Cloneable {
public long id;
public String name;
public B b = new B();
public Object clone() throws CloneNotSupportedException {
return super.clone();
}
public String toString() {
return "A [id=" + id + ",name=" + name + ",b=" + b + "]";
}
}
class B implements Cloneable {
public long id;
public String name;
public Object clone() throws CloneNotSupportedException {
return super.clone();
}
public String toString() {
return "B [id=" + id + ",name=" + name + "]";
}
}
public class CloneTest {
/**
* @param args
* @throws CloneNotSupportedException
*/
public static void main(String[] args) throws CloneNotSupportedException {
// TODO Auto-generated method stub
A a1 = new A();
a1.id = 1;
a1.name = "aaa";
a1.b.id = 2;
a1.b.name = "bbb";
A a2 = (A) a1.clone();
a1.id = 3;
a1.name = "ccc";
a1.b.id = 4;
a1.b.name = "ddd";
System.out.println(a2);
System.out.println(a1);
}
}
5.运行结果
A [id=1,name=aaa,b=B [id=4,name=ddd]]
A [id=3,name=ccc,b=B [id=4,name=ddd]]