覆盖clone时需要实现Cloneable接口,Cloneable并没有定义任何方法。
那Cloneable的意义是什么?
如果一个类实现了Clonable,Object的clone方法就可以返回该对象的逐域拷贝,否则会抛出CloneNotSupportedException

通常,实现接口是为了表明类的行为。
而Cloneable接口改变了超类中protected方法的行为。
这是种非典型用法,不值得仿效。

好了,既然覆盖了clone方法,我们需要遵守一些约定:

  • x.clone() != x;
  • x.clone().getClass() = x.getClass();
  • x.clone().equals(x);

另外,我们必须保证clone结果不能影响原始对象的同时保证clone方法的约定。

比如下面这种情况,没有覆盖clone方法,直接得到super.clone()的结果:

  1. import java.util.Arrays;
  2. public class Stack implements Cloneable {
  3. private Object[] elements;
  4. private int size = 0;
  5. private static final int DEFAULT_INITIAL_CAPACITY = 16;
  6. public Stack() {
  7. this.elements = new Object[DEFAULT_INITIAL_CAPACITY];
  8. }
  9. public void push(Object e) {
  10. ensureCapacity();
  11. elements[size++] = e;
  12. }
  13. public Object pop() {
  14. if (size == 0)
  15. throw new EmptyStackException();
  16. Object result = elements[--size];
  17. elements[size] = null; // Eliminate obsolete reference
  18. return result;
  19. }
  20. public boolean isEmpty() {
  21. return size == 0;
  22. }
  23. // Ensure space for at least one more element.
  24. private void ensureCapacity() {
  25. if (elements.length == size)
  26. elements = Arrays.copyOf(elements, 2 * size + 1);
  27. }
  28. }

结果可想而知,clone结果的elements和原始对象的elements引用同一个数组。

既然如此,覆盖clone方法,并保证不会伤害到原始对象:

  1. @Override
  2. public Stack clone() {
  3. try {
  4. Stack result = (Stack) super.clone();
  5. result.elements = elements.clone();
  6. return result;
  7. } catch (CloneNotSupportedException e) {
  8. throw new AssertionError();
  9. }
  10. }

虽然把elements单独拿出来clone了一遍,但这种做法的前提是elements不是final。
其实再正常不过,clone无法和引用可变对象的不可变field兼容。

如果数组的元素是引用类型,当某个元素发生改变时仍然会出现问题。
此处以Hashtable为例,Hashtable中的元素用其内部类Entry。

  1. private static class Entry<K,V> implements Map.Entry<K,V> {
  2. int hash;
  3. final K key;
  4. V value;
  5. Entry<K,V> next;
  6. protected Entry(int hash, K key, V value, Entry<K,V> next) {
  7. this.hash = hash;
  8. this.key = key;
  9. this.value = value;
  10. this.next = next;
  11. }
  12. //..
  13. }

如果像Stack例子中那样直接对elements进行clone,某个Entry发生变化时clone出来的Hashtable也随之发生变化。

于是Hashtable中如此覆盖clone:

  1. /**
  2. * Creates a shallow copy of this hashtable. All the structure of the
  3. * hashtable itself is copied, but the keys and values are not cloned.
  4. * This is a relatively expensive operation.
  5. *
  6. * @return a clone of the hashtable
  7. */
  8. public synchronized Object clone() {
  9. try {
  10. Hashtable<K,V> t = (Hashtable<K,V>) super.clone();
  11. t.table = new Entry[table.length];
  12. for (int i = table.length ; i-- > 0 ; ) {
  13. t.table[i] = (table[i] != null)
  14. ? (Entry<K,V>) table[i].clone() : null;
  15. }
  16. t.keySet = null;
  17. t.entrySet = null;
  18. t.values = null;
  19. t.modCount = 0;
  20. return t;
  21. } catch (CloneNotSupportedException e) {
  22. // this shouldn't happen, since we are Cloneable
  23. throw new InternalError();
  24. }
  25. }

鉴于clone会导致诸多问题,有两点建议:

  • 不要扩展Cloneable接口
  • 为继承而设计的类不要实现Cloneable接口

Java - 谨慎覆盖clone的更多相关文章

  1. Effective Java —— 谨慎覆盖clone

    本文参考 本篇文章参考自<Effective Java>第三版第十三条"Always override toString",在<阿里巴巴Java开发手册>中 ...

  2. Java - 谨慎覆盖equals

    平时很难遇到需要覆盖equals的情况. 什么时候不需要覆盖equals? 类的每个实例本质上是唯一的,我们不需要用特殊的逻辑值来表述,Object提供的equals方法正好是正确的. 超类已经覆盖了 ...

  3. Effective Java 之-----谨慎的覆盖clone方法

    1.概述 如果clone方法返回一个由构造器创建的对象,它就得到有错误的类.因此,如果覆盖了非final类中的clone方法,则应该返回一个通过调用super.clone得到的对象.如果类的所有超类都 ...

  4. Item 11 谨慎地覆盖Clone

    1.进行浅拷贝时,只是复制原始数据类型的值,则可以通过覆盖Clone方法来达到.另外,在进行浅拷贝的时候,还要注意,成员对象中不应该要有引用类型,如果有引用类型,那么,进行了浅拷贝之后,两个对象将会共 ...

  5. 第11条:谨慎地覆盖clone

    Clone提供一种语言之外的机制:无需调用构造器就可以创建对象. 它的通用约定非常弱: 创建和返回该对象的一个拷贝.这个拷贝的精确含义取决于该对象的类.一般含义是,对于任何对象x,表达式x.clone ...

  6. 详解Java中的clone方法

    详解Java中的clone方法 参考:http://blog.csdn.net/zhangjg_blog/article/details/18369201/ 所谓的复制对象,首先要分配一个和源对象同样 ...

  7. 浅拷贝和深拷贝(谈谈java中的clone)

    clone顾名思义就是复制, 在Java语言中, clone方法被对象调用,所以会复制对象.所谓的复制对象,首先要分配一个和源对象同样大小的空间,在这个空间中创建一个新的对象.那么在java语言中,有 ...

  8. java克隆对象clone()的使用方法和作用

    转自:997.html">http://www.okrs.cn/blog/news/?997.html 内容摘要 若需改动一个对象,同一时候不想改变调用者的对象.就要制作该对象的一个本 ...

  9. 深入理解Java中的Clone与深拷贝和浅拷贝

    1.Java对象的创建 clone顾名思义就是复制, 在Java语言中, clone方法被对象调用,所以会复制对象.所谓的复制对象,首先要分配一个和源对象同样大小的空间,在这个空间中创建一个新的对象. ...

随机推荐

  1. [Perl]Can't link/include C library 'ft2build.h', 'freetype', aborting.

    原文:http://www.code-by.org/viewtopic.php?f=60&t=284 错误提示 Font-FreeType-0.07>perl Makefile.PL B ...

  2. 题解 UVA11300 【Spreading the Wealth】

    环形均分纸牌问题应该不少人都很熟悉了,而且题解区写的也比较全了...... 我这篇题解主要是介绍一个新的STL--nth_element 以及解答几个其他题解里面有应用但是没有注释的问题.(比如说我第 ...

  3. IO模型《六》IO模型比较分析

    IO模型比较分析 到目前为止,已经将四个IO Model都介绍完了.现在回过头来回答最初的那几个问题:blocking和non-blocking的区别在哪,synchronous IO和asynchr ...

  4. OCP 12c最新考试原题及答案(071-8)

    8.(5-4) choose the best answer:You need to produce a report where each customer's credit limit has b ...

  5. Heap堆

    #pragma once#include<iostream>using namespace std;#include<vector> template<class T&g ...

  6. 1.Java Spring MVC入门 安装

    Spring 下载地址: 4.3.6.RELEASE/ 25-Jan-2017 14:05 - http://repo.spring.io/release/org/springframework/sp ...

  7. chrome cpu占用100%

    参考原文地址:https://stackoverflow.com/questions/20276097/chrome-devtools-100-cpu 问题描述,chrome打开devtools开发者 ...

  8. P3952 时间复杂度

    P3952 时间复杂度 题目描述 小明正在学习一种新的编程语言 A++,刚学会循环语句的他激动地写了好多程序并 给出了他自己算出的时间复杂度,可他的编程老师实在不想一个一个检查小明的程序, 于是你的机 ...

  9. Python3学习札记

    1.- (按位取反) x的按位取反结果为-(x+1)   e.g. -5输出-6 更多细节,阅:http://stackoverflow.com/a/11810203 2.DocString约定 为一 ...

  10. 条目二十四《当效率至关重要时,请在map::operator[]与map::insert之间谨慎做出选择》

    条目二十四<当效率至关重要时,请在map::operator[]与map::insert之间谨慎做出选择> 当效率至关重要时,应该在map::operator[]和map::insert之 ...