JAVA编程思想读书笔记(四)--对象的克隆

时间:2023-03-08 17:47:36

接上篇JAVA编程思想读书笔记(三)--RTTI

No1:

类的克隆

public class MyObject implements Cloneable {
int i; public MyObject(int i) {
this.i = i;
} @Override
protected Object clone() {
Object o = null;
try {
o = super.clone();
} catch (CloneNotSupportedException e) {
System.out.println("MyObject can't clone");
}
return o;
} @Override
public String toString() {
return Integer.toString(i);
}
}
public class LocalCopy {
public static void main(String args[]) {
check1();
check2();
} static MyObject g(MyObject v) {
v.i++;
return v;
} static MyObject f(MyObject v) {
v = (MyObject) v.clone();
v.i++;
return v;
} private static void check1() {
MyObject a = new MyObject(11);
MyObject b = g(a);
if (a == b) {
System.out.println("a==b");
} else {
System.out.println("a!=b");
}
System.out.println("a=" + a);
System.out.println("b=" + b);
} private static void check2() {
MyObject c = new MyObject(47);
MyObject d = f(c);
if (c == d) {
System.out.println("c==d");
} else {
System.out.println("c!=d");
}
System.out.println("c=" + c);
System.out.println("d=" + d);
}
}

输出结果

a==b
a=12
b=12
c!=d
c=47
d=48

若想克隆一个类,必须继承Cloneable,并且重写clone方法

g()演示的是按引用传递,它会修改外部对象,并返回对那个外部对象的一个引用。

f()是对自变量进行克隆,所以将其分离出来,并让原来的对象保持独立,甚至能返回指向这个新对象的一个句柄,而且不会对原来的对象产生任何副作用。

No2:

public class Snake implements Cloneable {
private Snake next;
private char c; Snake(int i, char x) {
c = x;
if (--i > 0) {
next = new Snake(i, (char) (x + 1));
}
} void increment() {
c++;
if (next != null) {
next.increment();
}
} @Override
public String toString() {
String s = ":" + c;
if (next != null) {
s += next.toString();
}
return s;
} @Override
protected Object clone() {
Object o = null;
try {
o = super.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return o;
} public static void main(String args[]) {
Snake s = new Snake(5, 'a');
System.out.println("s=" + s);
Snake s2 = (Snake) s.clone();
System.out.println("s2="+s2);
s.increment();
System.out.println("after s.increment:");
System.out.println("s="+s);
System.out.println("s2="+s2);
}
}

输出结果

s=:a:b:c:d:e
s2=:a:b:c:d:e
after s.increment:
s=:b:c:d:e:f
s2=:a:c:d:e:f

这个还需要再研究

No3:

深层复制

public class CloneA implements Cloneable {
private int i; public CloneA(int i) {
this.i = i;
} @Override
protected Object clone() {
Object o = null;
try {
o = super.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return o;
}
}
public class CloneB implements Cloneable {
private int j; public CloneB(int j) {
this.j = j;
} @Override
protected Object clone() {
Object o = null;
try {
o = super.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return o;
}
}
public class CloneAB implements Cloneable {
private CloneA cloneA;
private CloneB cloneB; public CloneAB(int i, int j) {
cloneA = new CloneA(i);
cloneB = new CloneB(j);
} @Override
protected Object clone() {
CloneAB o = null;
try {
o = (CloneAB) super.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
o.cloneA = (CloneA) o.cloneA.clone();
o.cloneB = (CloneB) o.cloneB.clone();
return o;
}
}
public class TextAB {
public static void main(String args[]) {
CloneAB ab = new CloneAB(11, 22);
CloneAB r = (CloneAB) ab.clone();
}
}

CloneAB由CloneA和CloneB两个对象合成,为了对其进行深层复制,必须同时克隆里面两个对象的句柄。

No4:

public class A {}

public class B {}

public class C extends A implements Cloneable {
@Override
protected Object clone() {
Object o = null;
try {
o = super.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return o;
}
} public class D extends C {}
public class TextABCD {
public static void main(String args[]) {
A a = new A();
B b = new B();
C c = new C();
D d = new D(); c.clone();
d.clone();
}
}

在添加克隆能力之前,编译器会阻止我们的克隆尝试。一旦在C类中添加了克隆能力,那么C及它的所有后代都可以克隆。

No5:

public class Ordinary {}
public class WrongClone extends Ordinary {

    @Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
} public class IsCloneable extends Ordinary implements Cloneable {
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
public class NoMore extends IsCloneable {
@Override
protected Object clone() throws CloneNotSupportedException {
throw new CloneNotSupportedException();
}
}
final class ReallyNoMore extends NoMore {}

public class TryMore extends NoMore {
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
} public class BackOn extends NoMore {
private BackOn duplicate(BackOn b) {
return new BackOn();
} public Object clone() {
return duplicate(this);
}
}
public class CheckCloneable {
static Ordinary tryToClone(Ordinary ord) {
String id = ord.getClass().getName();
Ordinary x = null;
if (ord instanceof Cloneable) {
try {
System.out.println("Attempting " + id);
x = (Ordinary) ((IsCloneable) ord).clone();
System.out.println("Cloned " + id);
} catch (CloneNotSupportedException e) {
System.out.println("Could not clone " + id);
}
}
return x;
} public static void main(String args[]) {
Ordinary[] ord = {new IsCloneable(),
new WrongClone(),
new NoMore(),
new TryMore(),
new BackOn(),
new ReallyNoMore(),
};
Ordinary x = new Ordinary();
for (int i = 0; i < ord.length; i++) {
tryToClone(ord[i]);
}
}
}

输出结果

Attempting IsCloneable
Cloned IsCloneable
Attempting NoMore
Could not clone NoMore
Attempting TryMore
Could not clone TryMore
Attempting BackOn
Cloned BackOn
Attempting ReallyNoMore
Could not clone ReallyNoMore

WrongClone没有继承Cloneable,所以不能克隆

IsCloneable是标准写法,可以克隆

NoMore直接抛异常,所以不能克隆

ReallyNoMore是final类,所以不能克隆

TryMore继承了NoMore直接抛异常,所以不能克隆

BackOn自己生成一个新对象,算是特殊的克隆

总之,如果希望一个类能够克隆

(1)实现Cloneable接口

(2)覆盖clone()

(3)在自己的clone()中调用super.clone()

(4)在自己的clone()中捕获违例

No6:

String的修改是生成了一个新的String对象,包括“+"的使用