接上篇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对象,包括“+"的使用

JAVA编程思想读书笔记(四)--对象的克隆的更多相关文章

  1. <Java编程思想>读书笔记(1)-对象导论、一切都是对象

    1.面向对象编程:OOP (Object-oriented Programming) 2.Alan Kay 总结的面向对象语言5个基本特性: 1) 万物皆为对象 2) 程序是对象的集合,他们通过发送消 ...

  2. JAVA编程思想读书笔记(五)--多线程

    接上篇JAVA编程思想读书笔记(四)--对象的克隆 No1: daemon Thread(守护线程) 参考http://blog.csdn.net/pony_maggie/article/detail ...

  3. JAVA编程思想读书笔记(二)--容器

    接上篇JAVA编程思想读书笔记(一) 第八章.对象的容纳 No1: java提供了四种类型的集合类:Vector(矢量).BitSet(位集).Stack(堆栈).Hashtable(散列表) No2 ...

  4. JAVA编程思想读书笔记(三)--RTTI

    接上篇JAVA编程思想读书笔记(二) 第十一章 运行期类型判定 No1: 对于作为程序一部分的每个类,它们都有一个Class对象.换言之,每次写一个新类时,同时也会创建一个Class对象(更恰当的说, ...

  5. Java编程思想读书笔记之内部类

    现在是够懒得了,放假的时候就想把这篇笔记写出来,一直拖到现在,最近在读<Java编程思想>,我想会做不止这一篇笔记,因为之前面试的时候总会问道一些内部类的问题,那这本书的笔记就从内部类开始 ...

  6. Java编程思想读书笔记(一)【对象导论】

    2018年1月7日15:45:58 前言 作为学习Java语言的经典之作<Java编程思想>,常常被人提起.虽然这本书出版十年有余,但是内容还是很给力的.很多人说这本书不是很适合初学者,我 ...

  7. Java编程思想读书笔记(二)【一切都是对象】

    begin 2018年1月9日17:06:47 第二章 一切都是对象 Java语言假设我们只进行面向对象的程序设计. 2.1 用引用操纵对象 每种编程语言都有自己的操纵内存元素的方式 操纵内存元素的方 ...

  8. Java编程思想读书笔记

    声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...

  9. Java编程思想读书笔记之一切皆对象

    一切皆对象 Java程序运行时,数据保存到哪里 寄存器 这是最快的保存区域,因为它位于和其他所有保存方式不同的地方:处理器内部.然而,寄存器的数量十分有限,所以寄存器是根据需要由编译器分配.我们对此没 ...

随机推荐

  1. [DeeplearningAI笔记]序列模型2.10词嵌入除偏

    5.2自然语言处理 觉得有用的话,欢迎一起讨论相互学习~Follow Me 2.10词嵌入除偏 Debiasing word embeddings Bolukbasi T, Chang K W, Zo ...

  2. Vue.js随笔四(方法的声明和使用)

    1.首先你需要新建路由,这个就不多说了 2.然后在你的新的.vue里面需要如下所示的添加methods:{方法},然后按钮的里面你会看到v-on:click,这就是点击这个按钮会触发的动作,这个就是触 ...

  3. CSS中em,rem的区别

    首先这两个单位一般用在移动端 不太清楚得求证  再记录 1.em w3cschool中给出css中尺寸单位如下: 单位 描述 % 百分比 in 英寸 cm 厘米 mm 毫米 em 1em 等于当前的字 ...

  4. Spring整合JMS(四)——事务管理(转)

    *注:别人那复制来的 Spring提供了一个JmsTransactionManager用于对JMS ConnectionFactory做事务管理.这将允许JMS应用利用Spring的事务管理特性.Jm ...

  5. SpringCloud(二)注册服务提供者搭建

    上文已经写了如何去搭建注册中心,仅有注册中心是远远不够的,所以我们需要注册到注册中心并提供服务的节点,这里称为注册服务提供者 前提 阅读上文,并成功搭建注册中心,环境无需改变 项目搭建 这里我们需要新 ...

  6. GlusterFS + lagstash + elasticsearch + kibana 3 + redis日志收集存储系统部署 01

    因公司数据安全和分析的需要,故调研了一下 GlusterFS + lagstash + elasticsearch + kibana 3 + redis 整合在一起的日志管理应用: 安装,配置过程,使 ...

  7. 从零搭建SSM框架(三)SSM框架整合

    整合思路 1.Dao层: Mybatis的配置文件:SqlMapConfig.xml 不需要配置任何内容,需要有文件头.文件必须存在. applicationContext-dao.xml: myba ...

  8. 天梯赛 L2-006 树的遍历 (二叉树)

    给定一棵二叉树的后序遍历和中序遍历,请你输出其层序遍历的序列.这里假设键值都是互不相等的正整数. 输入格式: 输入第一行给出一个正整数N(<=30),是二叉树中结点的个数.第二行给出其后序遍历序 ...

  9. php常用函数——字符串函数

    php常用函数——字符串函数

  10. OGG生成数据定义文件的参数NOEXTATTR

    ./defgen paramfile ./dirprm/jzjj.prm NOEXTATTR In OGG 11.2, there is a new parameter NOEXTATTR. This ...