第九十六节 Java面向对象设计 - Java Object.Clone方法

Java面向对象设计 - Java Object.Clone方法

Java不提供克隆(复制)对象的自动机制。

克隆对象意味着逐位复制对象的内容。

要支持克隆操作,请在类中实现clone()方法。

Object类中的clone()方法的声明如下:

protected  Object clone()  throws   CloneNotSupportedException

clone()方法声明为protected。因此,我们不能从客户端代码调用它。以下代码无效:

Object obj  = new Object();
Object clone = obj.clone(); // Error. Cannot  access protected clone() method

我们需要在类中声明clone()方法public克隆类的对象。

它的返回类型是Object。这意味着您将需要转换clone()方法的返回值。

假设MyClass是可克隆的。克隆代码将如下所示:

MyClass mc  = new MyClass();
MyClass clone = (MyClass)mc.clone(); // Need to use  a  cast

Object类中的clone()方法会抛出CloneNotSupportedException。

要调用clone()方法,我们需要将调用放在try-catch块中,或者重新抛出异常。

例子

以下代码显示了如何实现克隆方法。

class MyClass implements Cloneable {
  private double value;

  public MyClass(double value) {
    this.value = value;
  }

  public void setValue(double value) {
    this.value = value;
  }

  public double getValue() {
    return this.value;
  }

  public Object clone() {
    MyClass copy = null;
    try {
      copy = (MyClass) super.clone();
    } catch (CloneNotSupportedException e) {
      e.printStackTrace();
    }
    return copy;
  }
}

public class Main {
  public static void main(String[] args) {
    MyClass dh = new MyClass(100.00);

    MyClass dhClone = (MyClass) dh.clone();

    System.out.println("Original:" + dh.getValue());
    System.out.println("Clone :" + dhClone.getValue());

    dh.setValue(200.00);
    dhClone.setValue(400.00);

    System.out.println("Original:" + dh.getValue());
    System.out.println("Clone :" + dhClone.getValue());
  }
}

上面的代码生成以下结果。

Original:100.0
Clone :100.0
Original:200.0
Clone :400.0

例2

以下代码不从clone方法返回对象类型,该方法仅在Java 5或更高版本中编译。

class MyClass  implements Cloneable  {
    public MyClass clone()  { 
       Object copy  = null;
       return  (MyClass)copy;
    }
}

下面的代码展示了如何做浅层克隆。

class MyClass implements Cloneable {
  private double value;

  public MyClass(double value) {
    this.value = value;
  }

  public void setValue(double value) {
    this.value = value;
  }

  public double getValue() {
    return this.value;
  }

  public Object clone() {
    MyClass copy = null;
    try {
      copy = (MyClass) super.clone();
    } catch (CloneNotSupportedException e) {
      e.printStackTrace();
    }
    return copy;
  }
}

class ShallowClone implements Cloneable {
  private MyClass holder = new MyClass(0.0);

  public ShallowClone(double value) {
    this.holder.setValue(value);
  }

  public void setValue(double value) {
    this.holder.setValue(value);
  }

  public double getValue() {
    return this.holder.getValue();
  }

  public Object clone() {
    ShallowClone copy = null;
    try {
      copy = (ShallowClone) super.clone();
    } catch (CloneNotSupportedException e) {
      e.printStackTrace();
    }
    return copy;
  }
}

public class Main {
  public static void main(String[] args) {
    ShallowClone sc = new ShallowClone(100.00);
    ShallowClone scClone = (ShallowClone) sc.clone();

    System.out.println("Original:" + sc.getValue());
    System.out.println("Clone :" + scClone.getValue());

    sc.setValue(200.00);

    System.out.println("Original:" + sc.getValue());
    System.out.println("Clone :" + scClone.getValue());
  }
}

上面的代码生成以下结果。

例3

ShallowClone类的clone()方法中的代码与MyClass类的clone()方法相同。

当ShallowClone类使用super.clone()调用Object类的clone()方法时,它会接收自身的浅拷贝。也就是说,它与其克隆共享其实例变量中使用的DoubleHolder对象。

在深层克隆中,您需要克隆对象的所有引用实例变量引用的所有对象。

class MyClass implements Cloneable {
  private double value;

  public MyClass(double value) {
    this.value = value;
  }

  public void setValue(double value) {
    this.value = value;
  }

  public double getValue() {
    return this.value;
  }

  public Object clone() {
    MyClass copy = null;
    try {
      copy = (MyClass) super.clone();
    } catch (CloneNotSupportedException e) {
      e.printStackTrace();
    }
    return copy;
  }
}

class DeepClone implements Cloneable {
  private MyClass holder = new MyClass(0.0);

  public DeepClone(double value) {
    this.holder.setValue(value);
  }

  public void setValue(double value) {
    this.holder.setValue(value);
  }

  public double getValue() {
    return this.holder.getValue();
  }
  public Object clone() {
    DeepClone copy = null;
    try {
      copy = (DeepClone) super.clone();
      copy.holder = (MyClass) this.holder.clone();
    } catch (CloneNotSupportedException e) {
      e.printStackTrace();
    }
    return copy;
  }
}

public class Main {
  public static void main(String[] args) {
    DeepClone sc = new DeepClone(100.00);
    DeepClone scClone = (DeepClone) sc.clone();

    System.out.println("Original:" + sc.getValue());
    System.out.println("Clone :" + scClone.getValue());

    sc.setValue(200.00);

    System.out.println("Original:" + sc.getValue());
    System.out.println("Clone :" + scClone.getValue());
  }
}

上面的代码生成以下结果。

  • 26
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: clone方法:是Object类中的一个方法,用于创建并返回一个对象的副本。这个方法可以被子类重写,以实现深拷贝或浅拷贝。 标识接口:是一个没有方法的接口,只是用于标识一个类是否具有某种特定的能力或属性。例如,Serializable接口就是一个标识接口,用于标识一个类可以被序列化。 深拷贝:是指在拷贝对象时,不仅拷贝对象本身,还要拷贝对象所引用的其他对象。这样可以保证拷贝后的对象与原对象完全独立,互不影响。相对于浅拷贝,深拷贝需要更多的时间和空间。 ### 回答2: clone方法JavaObject类提供的一个方法,用于创建当前对象的一个副本。通过实现Cloneable接口并重写clone()方法,可以实现克隆对象的功能。这种克隆方式又被称为浅拷贝,因为只是将对象的引用复制了一份。 标识接口是一种没有方法和属性的接口,只起到标识一个类实现了该接口的作用。Java中常见的标识接口有Serializable、Cloneable接口等。因为标识接口中没有方法和属性,所以实现标识接口的类只需要在类声明上实现该接口即可。 深拷贝是与浅拷贝相对的一种拷贝方式,它不仅复制对象的引用,而是将对象的所有属性按照值复制一份。在Java中实现深拷贝可以通过序列化和反序列化实现,或者通过递归复制对象的所有属性。 对于Cloneable接口的实现,需要注意以下几点: 1.需要实现Cloneable接口,否则会在调用clone()方法时抛出CloneNotSupportedException异常。 2.需要重写clone()方法,因为Object类中的clone()方法是受保护的,只有实现了Cloneable接口并重写了clone()方法的类才能调用clone()方法。 3.重写的clone()方法需要调用父类的clone()方法并强制转换类型,否则克隆出来的对象类型会是Object类型。 在深拷贝的实现中,可以使用序列化和反序列化的方式: 1.对象序列化:将对象序列化后存储到内存或磁盘中。 2.对象反序列化:从内存或磁盘中读取序列化后的对象并反序列化为Java对象。在反序列化时会完全复制对象的属性和方法。 使用序列化和反序列化的方式实现深拷贝需要注意以下几点: 1.被拷贝的对象需要实现Serializable接口。 2.在序列化和反序列化时需要注意对象中包含的引用类型对象。 3.被拷贝的对象和拷贝出的对象之间是相互独立的,修改其中一个对象不会影响到另一个对象。 综上所述,clone方法、标识接口、深拷贝都是Java面向对象中重要的概念,它们可以实现对象的复制、序列化和反序列化等操作。对于Java开发者来说,深入理解这些概念并掌握实现方法是十分必要的。 ### 回答3: 面向对象是程序设计的一种重要思想,其中的一些基础概念及特性如克隆方法(clone)、标识接口以及深拷贝是程序设计中的重要内容。下面对这些概念进行详细介绍。 克隆方法Java语言中用来复制对象的方法。在Java中,要实现对象的复制,可以使用clone方法,该方法Object类的一个方法,意为克隆对象。它被称为“浅拷贝”,因为它只是复制了对象本身以及其所有引用类型的成员变量的引用,而不会复制引用类型的成员变量本身。因此,在使用clone方法时需要注意,被复制对象中的引用类型成员变量可能会被多个对象共享,引起错误。 标识接口是一种没有方法的接口。它仅用来给某些类打上标识,告诉编译器这些类有一些特殊的功能需要支持。Java语言中的Serializable和Cloneable就是两个常用的标识接口。Serializable用于标识一个类可以被序列化,而Cloneable用于标识一个类可以被克隆。 深拷贝是相对于“浅拷贝”而言的。它不仅复制了对象本身以及其引用类型的成员变量的引用,还会复制引用类型的成员变量本身。通常深拷贝需要在代码中手动实现,而且代码复杂度高。Java中提供了一种编写简单安全的深拷贝代码的方法,即通过实现Serializable接口进行对象序列化和反序列化,这样能够在内存中创建一个新的对象,它包含了与原始对象相同的值。 在实际编写面向对象程序时,深拷贝、克隆方法以及标识接口都是非常重要的概念。深刻理解这些概念的含义,可以帮助程序员更加灵活地运用这些手段,提高面向对象编程技术的应用水平。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值