如何在java中克隆对象

时间:2011-02-21 13:18:41

标签: java

我想创建一个具有相同父类的对象列表/数组,然后我将它用于参考。但我不知道如何克隆这些对象来制作一个新对象。

这是示例

BigFoo a;
SmallFoo b;
ChickenFoo c;
List<Foo> foos;
foos.add(a);
foos.add(b);
foos.add(c);

Foo foo = foos.get(1).clone();

但在Java中我发现默认函数中没有克隆函数。我想知道这是如何实现的?

3 个答案:

答案 0 :(得分:5)

我建议您阅读有关如何实现/公开clone()方法的内容。 http://download.oracle.com/javase/tutorial/java/IandI/objectclass.html

答案 1 :(得分:4)

一般建议:使用复制构造函数。事实上,只有一个类本身知道如何创建自己的克隆。没有类可以克隆另一个类的实例。这个想法是这样的:

public class Foo {
  public List<Bar> bars = new ArrayList<Bar>();
  private String secret;

  // Copy constructor
  public Foo(Foo that) {
    // new List
    this.bars = new ArrayList<Bar>();

    // add a clone of each bar (as an example, if you need "deep cloning")
    for (Bar bar:that.bars) {
      this.bars.add(new Bar(bar));
    }

    // clone the secret value
    this.secret = new String(that.secret);
  }

  // ...

}

因此,如果我们要克隆foo,我们只需根据foo创建一个新的:

Foo clonedFoo = new Foo(foo);

这是克隆实例的推荐方法。


复制构造函数适用于继承。考虑一个子类

 public ChildFoo extends Foo {

   private int key;

   public ChildFoo(ChildFoo that) {
     super(that);
     this.key = that.key;
   }
 }

Foo有一个复制构造函数,而ChildFoo只是从它自己的复制构造函数中调用它。

您的示例是可能的,但不可取。会发生什么:

 Foo a = new Foo();
 ChildFoo b = new ChildFoo(a);  

这将要求 ChildFoo上的构造函数,如:

 public ChildFoo(Foo that) {
     // call the copy constructor of Foo -> no problem
     super(that);

     // but how to initialize this.key? A Foo instance has no key value!
     // Maybe use a default value?
     this.key = 0;
 }

技术上不是挑战,但b不是a的克隆,因为对象的类型不同。所以这个(你的例子)不是克隆

答案 2 :(得分:1)

一种简单的方法是使用json映射器(Jackson或Gson)并将对象写为字符串,然后使用字符串创建克隆对象。