Clone提供一种语言之外的机制:无需调用构造器就可以创建对象。

它的通用约定非常弱:

创建和返回该对象的一个拷贝。这个拷贝的精确含义取决于该对象的类。一般含义是,对于任何对象x,表达式x.clone() != x 将会是true,并且,表达式x.clone().getClass() == x.getClass() 将会是true,但这些不是绝对的要求,通常情况下,表达式 x.clone().equals(x) 将会是true,这也不是一个绝对的要求,拷贝对象往往是创建它的类的一个新实例,但它同时也会要求拷贝内部的数据结构。

如果类的每个域包含一个基本类型的值,或者包含一个指向不可变对象的引用,那么被返回的对象则正是所需要的对象,如

public class PhoneNumber implements Cloneable{private final int areaCode;private final int prefix;private final int lineNumber;public PhoneNumber(int areaCode, int prefix, int lineNumber) {rangeCheck(areaCode, 999, "area code");rangeCheck(prefix, 999, "prefix");rangeCheck(lineNumber, 9999, "line number");this.areaCode = areaCode;this.prefix = prefix;this.lineNumber = lineNumber;}private static void rangeCheck(int arg, int max, String name) {if(arg < 0 || arg > max) {throw new IllegalArgumentException(name + ": " + arg);}}@Overridepublic boolean equals(Object o) {if(o == this)return true;if(!(o instanceof PhoneNumber))return false;PhoneNumber pn = (PhoneNumber)o;return pn.lineNumber == lineNumber&& pn.prefix == prefix&& pn.areaCode == areaCode;}@Overridepublic PhoneNumber clone() {try {return (PhoneNumber) super.clone();} catch(CloneNotSupportedException e) {throw new AssertionError();}}
}

不可变类包括String,Integer,Short等包装类。

而如果对象中包含的域引用了可变的对象,如:

public class Stack {private Object[] elements;private int size = 0;private static final int DEFAULT_INITAL_CAPACITY = 16;public Stack() {elements = new Object[DEFAULT_INITAL_CAPACITY];}public void push(Object e) {ensureCapacity();elements[size++] = e;}public Object pop() {if(size == 0) {throw new EmptyStackException();}Object result = elements[--size];elements[size] = null;return result;}private void ensureCapacity() {if(elements.length == size)elements = Arrays.copyOf(elements, 2 * size + 1);}
}

那么我们就要对引用对象进行深度赋值。
这里应该:

@Override
public Stack clone() {try {Stack result = (Stack) super.clone();result.elements = elements.clone();return result;} catch (CloneNotSupportedException e) {throw new AssertionError();}
}

如果是Integer等不可变对象,则不用进行clone,如果是数组,包括普通类型数组,或者包装类型数组,如int[],Integer[]则都要进行clone。

cloneable的问题导致我们不应该扩展这个接口,为了继承而设计的类也不应该实现这个接口,由于它具有这么多缺点,专家级的程序员从来不去覆盖clone方法, 也从来不去调用它,除非拷贝数组。

对于一个专门为继承而设计的类,如果未能提供行为良好的受保护clone方法,它的子类就不可能实现Cloneable接口。

要想实现真正的深复制而不希望实现Cloneable接口,可行的办法是实现Serializable接口,通过对象的序列化和反序列化实现克隆,可以实现真正的深度克隆。不过因为序列化和反序列化,会造成一定的性能损失。

Effective Java之谨慎地覆盖clone(十一)相关推荐

  1. 谨慎的覆盖clone方法

    2019独角兽企业重金招聘Python工程师标准>>> 说在前面 有些专家级程序员干脆从来不去覆盖clone方法,也从来不去调用它,除非拷贝数组. 其他方式 可以提供一个构造函数或者 ...

  2. Effective Java之谨慎地使用本地方法(五十四)

    本地方法,是指本地程序设计语言(c,或者c++)来编写的特殊方法. 本地方法在本地语言中可移植性任意的计算任务,并且返回到java程序语言. 为什么说谨慎使用本地方法? 1.如果调用了一个本地方法,那 ...

  3. Effective Java之谨慎地实现Serializable(七十四)

    1.序列化的含义和作用 序列化用来将对象编码成字节流,反序列化就使将字节流编码重新构建对象. 序列化实现了对象传输和对象持久化,所以它能够为远程通信提供对象表示法,为JavaBean组件提供持久化数据 ...

  4. Effective Java~45. 谨慎使用Stream

    在 Java 8 中添加了 Stream API,以简化顺序或并行执行批量操作的任务. 该 API 提供了两个关键的抽象:流(Stream),表示有限或无限的数据元素序列,以及流管道 (stream ...

  5. Effective Java(第3版) 90条经验法则

    目录 第2章 创建和销毁对象 第1条:用静态工厂方法替代构造器 第2条:遇到多个构造器参数时要考虑使用构建器 例子 使用 第3条:用私有构造器或者枚举类型强化Singleton属性 例子 使用 第4条 ...

  6. Effective java -- 2 对于所有对象都通用到方法

    第八条:覆盖equals时请遵守通用约定 什么时候需要覆盖equals方法?类具有自己的逻辑相等概念,并且父类的equals方法不能满足需要. 重写equals时需要遵循一下约定: 自反性:非null ...

  7. 《Effective Java》读书笔记

    引言 1 代码应该被重用 ,而不是被拷贝. 2 错误应该尽早被检测出来,最好是在编译时刻. 3 接口.类.数组是引用类型(对象), 基本类型不是 第二章 创建和销毁对象 1 考虑用静态工厂方法代替构造 ...

  8. 《Effective Java》读书笔记,flutter游戏开发

    1 代码应该被重用 ,而不是被拷贝. 2 错误应该尽早被检测出来,最好是在编译时刻. 3 接口.类.数组是引用类型(对象), 基本类型不是 第二章 创建和销毁对象 1 考虑用静态工厂方法代替构造器. ...

  9. 《Effective Java》中文版第3版 读书笔记

    评论中有电子档资源哦  ^_^ 第1章引言 第2章创建和销毁对象 第1条:用静态工厂方法代替构造器 静态工厂方法与构造器不同的第一大优势在于,它们有名称.  静态工厂方法与构造器不同的第二大优势在于, ...

最新文章

  1. 两个链表的第一个公共节点分析
  2. 年终盘点:2021年AI领域值得关注的十件事
  3. CactiEZ安装配置教程
  4. 各类web服务器 http access log获取用户和代理ip
  5. 2021 The 4th International Conference on Information and Computer Technologies Certificate
  6. excel中如何将时间戳转换为日期格式
  7. CF1479D Odd Mineral Resource
  8. python需要配置环境变量吗_python为什么会环境变量设置不成功
  9. 涂国旗(洛谷P3392题题解,Java语言描述)
  10. linux下载pycharm_django开发-使用pycharm进行远程开发
  11. 十、Shell脚本编程
  12. linux串口导致死机,Linux系统死机情况分析与处理方案介绍
  13. 计算机专业的大专大学规划,大学生计算机专业职业规划个人简历
  14. linux怎么卸载vsftpd软件,vsftpd配置详解之软件安装和卸载
  15. Android mc怎么和win10联机,我的世界手机版/win10版联机完美互通方法
  16. idea使用教程-安装
  17. Linux系统下下载Tomcat详细步骤。
  18. Arm开发板上使用ldd命令
  19. YOLO测试图片显示准确率值
  20. 如何设计好系统异常处理

热门文章

  1. 外挂学习之路(11)--- 背包数据的遍历
  2. 玩转Google开源C++单元测试框架Google Test系列(gtest)之六 - 运行参数
  3. Android QEMU 高速管道
  4. 使用 ortp 发送原始 H.264 码流
  5. 深入理解Linux调度子系统
  6. 深入理解 MySQL 索引底层原理
  7. 牛逼! IDEA 2020 要本土化,真的是全中文了!
  8. Java中已经存在了十几年的一个bug...
  9. 在公司的微服务上搞破坏真是太开心了
  10. 微博客户端播放器的演进之路