Java Clone

更多 Java 高级知识方面的文章,请参见文集《Java 高级知识》


首先看下面这一段代码:

Student s = new Student();
Student s1= s;
  • 如果是在 C++ 中,会调用拷贝构造函数,复制一份新的对象,由 s1 指向该新的对象
  • 如果是在 Java 中,s1 为引用,仍然指向 s 的内存区域,即同一份对象

在下面的代码中:
c1c2 刚开始分别指向两个不同的对象,c2 = c1; 使得 c2 指向 c1 的对象,因此 c2 原本指向的对象 B 失去引用,会被 GC 回收。

public class Clone_Test {
    public static void main(String[] args) {
        MyClone c1 = new MyClone("A");
        MyClone c2 = new MyClone("B");

        // c2 指向 c1 的对象
        // c2 原本指向的对象 B 失去引用,会被 GC 回收
        c2 = c1;
        c2.name = "C";

        System.out.println(c1.name); // 输出 C
        System.out.println(c2.name); // 输出 C
    }
}

class MyClone {
    public String name;

    public MyClone(String name) {
        this.name = name;
    }
}

Clone 的实现

实现 Cloneable 接口,并重写 clone() 方法。

关于 Cloneable 接口

  • Cloneable 是一个空接口。并不包含任何方法,更像是一个标记。
public interface Cloneable {
}
  • implements Cloneable 指示调用 clone() 方法时可以合法地对该对象进行按字段的复制

A class implements the <code>Cloneable</code> interface to indicate to the clone() method that it is legal for that method to make a field-for-field copy of instances of that class.

  • 如果不实现 Cloneable 接口,此时在调用clone() 方法时,会抛出异常 CloneNotSupportedException

关于 Clone 方法

  • clone() 方法是 Object 类中自带的 native 方法,参见 Java Object 类中有哪些方法
  • clone() 方法默认是浅复制
    • 对基础类型有用
    • 对引用类型只是复制引用,不是真正地复制对象
  • clone() 方法不会调用构造方法
  • 几个基本的比较:
x.clone() == x // false,因为创建出了一个新对象
x.clone().getClass() == x.getClass() // true
x.clone().equals(x) // 一般为false,取决于 clone() 方法的具体实现

例如将上述的代码重构为:

public class Clone_Test {
    public static void main(String[] args) {
        MyClone c1 = new MyClone("A");
        MyClone c2 = new MyClone("B");

        // c2 指向一个新对象
        // c2 原本指向的对象 B 失去引用,会被 GC 回收
        c2 = (MyClone) c1.clone();
        c2.name = "C";

        System.out.println(c1.name); // 输出 A
        System.out.println(c2.name); // 输出 C
    }
}

class MyClone implements Cloneable {
    public String name;

    public MyClone(String name) {
        this.name = name;
    }

    public Object clone() {
        MyClone c = null;

        try {
            c = (MyClone) super.clone();
        } catch (Exception e) {
        }

        return c;
    }
}

浅拷贝 VS 深拷贝

Object 类中自带的clone() 方法是浅拷贝,即 field-for-field copy。
如果某个对象中包含一些其他对象,例如 Stream,浅拷贝只会复制该 Stream 对象的引用,该 Stream 对象会被两个对象 c1c2 共享,可能会导致问题。

要想实现深拷贝,即对象中包含的其他对象也会复制出一份新的对象,可以通过如下方式:

  • 需要重写 clone() 方法,实现深拷贝
  • 通过序列化方式 ObjectOutputStream ObjectInputStream
  • 通过第三方库
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 在java中,如果需要有拷贝问题,都会使用到父类Object的Clone方法,能够为我们提供对象的拷贝方法,在使用...
    三木仔阅读 5,427评论 0 2
  • Java Clone 平时项目中用的也不多,今天来实践下Java的Clone。Clone主要分为“浅拷贝”与“深拷...
    昵称全尼马被注册了阅读 4,039评论 0 3
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 33,600评论 18 399
  • 11.Override clone judiciously 大意为 明智地重写clone方法 说到clone方法,...
    Mezereon阅读 4,179评论 0 3
  • java笔记第一天 == 和 equals ==比较的比较的是两个变量的值是否相等,对于引用型变量表示的是两个变量...
    jmychou阅读 5,442评论 0 3

友情链接更多精彩内容