【问题标题】:Initialize object A in java equal to another object B [duplicate]在java中初始化对象A等于另一个对象B [重复]
【发布时间】:2012-06-30 05:09:32
【问题描述】:

可能重复:
How do I copy an object in Java?

如何在 java 中初始化一个对象(比如 A)并将其初始成员值设置为等于第二个对象(比如 B)。初始化后我想修改A的成员而不修改B的成员。所以在A的初始化时我只想复制B的数据。这是如何以一种好的方式完成的??

【问题讨论】:

标签: java object pointers initialization


【解决方案1】:

克隆是一种简单的复制方式。如果您想做一些需要更多控制的事情,请创建您自己的方法来完全按照您的需要执行副本:

public MyType copy()
{
  MyType a = new MyType();
  // Initialize how you need to here, use the object this was called from if you'd like
  a.property = this.property;
  // etc.  
  return a;
}

这为您提供了更直接的控制,但需要更多时间来编写代码。如果克隆适合您的目的,请坚持下去。

编辑:我将根据您的 cmets 给出一个示例。

假设我们有以下类型:

TypeA: has the following member variables
int number = 5; // Default value built in by constructor.
int letter = 'x'; // Value is 'a' when constructed but has been changed.
ArrayList<TypeB> list = {b1, b2, b3} // All are initialized.

TypeB: has the following member variables
double decimal = 5.32
TypeC someObject = ...

TypeC: has some stuff, but we are going to ignore it.

现在,当我们要复制 TypeA 时,我们必须执行以下操作:

  1. 直接复制数字和字符,因为它们是值类型。
  2. 复制对 ArrayList 的引用,其中包含对某些 TypeB 的引用。

幸运的是,这些步骤很简单。

int copyNumber = this.number;
char copyChar = this.letter;
ArrayList<TypeB> copyList = this.list;
return new TypeA(copyNumber, copyChar, copyList);

现在假设有一个特定的构造函数接受这三个参数,但希望你明白这一点。

如果您只想获取值,而不是对 ArrayList 中所有 TypeB 的引用,这将变得很棘手。您必须遍历 ArrayList 并创建复制所有 ITS 值的新 TypeB(双精度和 TypeC 对象作为引用或值...)

简而言之,您想要的是更容易执行的副本。简单的赋值运算符复制原始类型的值和对象的引用。

【讨论】:

  • 好的,但是如果我要复制的对象包含其他对象作为方法?
  • @sjoerd999:您遇到了什么问题?我或许可以帮助你。
  • 好的,我来解释一下:我有一个对象A,它有其他对象作为成员。并且还具有数据结构作为成员(数组,列表等......)。因此,复制构造函数会有点困难,因为据我了解,您必须在原始数据类型级别复制所有内容。由于同样的原因,克隆也不起作用。那么该怎么办?还是我现在把这个看似简单的问题变得太难了?
  • 啊,所以问题变得有点棘手。您想将 references 复制到这些其他对象吗?还是您只对这些对象所代表的感兴趣?
  • 我想复制原始数据类型以及数组和列表的所有值。我想复制对作为对象 A 成员的对象的引用。
【解决方案2】:

这一切都取决于成员的类型。我举个例子:

class A
{
    public float value;
    public int[] anArray;

    public A(B b)
    {
        //primitive may be assigned directly.
        this.value = b.value;

        // other types different approaches:

        //copy the contents of the array
        this.anArray = new int[b.anArray.length];
        System.arraycopy(b.anArray, 0, this.anArray, 0, b.anArray.length);
    }
}

class B
{
    float value;
    int[] anArray;
    public B(int size)
    {
        this.value = 3f;
        this.anArray = new int[size];
        for (int i = size - 1; i >= 0; i--)
        {
            this.anArray[i] = i * 10;
        }
    }
}

B b = new B(5);
A a = new A(b);

【讨论】:

    【解决方案3】:

    你可以实现和使用克隆

    MyClass b = new MyClass();
    MyClass a = b.clone();
    

    注意:有些类是不可克隆的,或者有损坏的实现。例如只有当它们应该是深拷贝时才具有浅拷贝。

    如果类是Serializable,你可以序列化它并在内存中反序列化它。不完全是nice,但它有效。

    或者您可以创建自己的“复制”构造函数。

    【讨论】:

    • ...假设它实现了Cloneable,并且clone 在这个特定的实现中没有被破坏。
    【解决方案4】:

    一种可能的解决方案是在您的类上实现clone 方法并使用克隆,如下所示:

    MyClass a = new MyClass();
    MyClass b = a;
    

    您会注意到clone() 并不是真正的公共方法,因此您需要公开它。此外,您需要告诉 Java 您的对象是可克隆的(这样做是为了让您的类实现 Cloneable)。以下代码说明了这一点:

    public class MyClass implements Cloneable {
    
        @Override
        protected MyClass clone() throws CloneNotSupportedException {
            return (MyClass)super.clone();
        }
    
    }
    

    【讨论】:

      猜你喜欢
      • 2016-05-30
      • 1970-01-01
      • 2016-12-18
      • 2012-01-17
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多