Java - 反射 - 如何调用复制构造函数?

时间:2013-11-10 09:22:50

标签: java reflection copy-constructor

我有这段代码:

我要复制的课程:

public class NormalChair extends AbstractChair {
    protected int height;
    protected String name;
    public NormalChair() {
        super();
    }

    public NormalChair(String name, int height) {
        super(name, height);
    }


    // Copy constructor - getName() and getHeight() are defined in parent class.
    public NormalChair(NormalChair chair) {
      this(chair.getName(), chair.getHeight());
    } 
}

创建一些课程

public Object createObj(String cls_name,String param1,int param2){   返回Class.forName(cls_name).getConstructor(String.class,Integer.class).newInstance(param1,param2); }

然后我尝试使用以下方法复制该类的对象:

Object obj_to_copy = createObj("Classname", "name", 10);
String cls_name = obj_to_copy.getClass().getName();
Class.forName(cls_name).getConstructor(Object.class).newInstance(obj_to_copy);

我收到了这个错误:

Exception in thread "main" java.lang.NoSuchMethodException: test.NormalChair.<init>(java.lang.Object)
    at java.lang.Class.getConstructor0(Class.java:2800)
    at java.lang.Class.getConstructor(Class.java:1708)
    at test.ProductTrader.create(ProductTrader.java:57)
    at test.Test.main(Test.java:23)

所以我想我需要以某种方式调用复制构造函数,而不是将它的类型显示为Object?

P.S。 我也把这个例子简单化了。但实际上我不知道在运行之前需要复制哪个类,因此使用复制构造函数不应仅依赖于NormalChair class.

更新

我更新了我的问题,为了更清楚地说明当我复制对象之前,运行时,我不知道它需要复制哪个类。

3 个答案:

答案 0 :(得分:0)

由于某种原因,Java反射严格匹配类和方法签名。因此,为了找到匹配的构造函数,您需要使用Class.getDeclaredConstructors()枚举可用的构造函数并找到匹配的构造函数。

我编写了一个小型库来简化任务,这里有一个匹配类的方法:HavingMethodSignature

如果您有兴趣,请按以下方式create a new instance使用此库:

Object o = OpenBean.newInstance(Class.forName(cls_name));

答案 1 :(得分:0)

为什么要在Java中使用复制构造函数呢?有一种在Java中复制对象的标准方法:简单地克隆它。如果默认克隆不相关,请覆盖clone()方法。

您只需编写obj.clone()即可获得副本。

有关详细信息,请参阅Java documentation of clone()

答案 2 :(得分:0)

如果您可以假设复制构造函数接受同一个类的对象,您可以执行以下操作:

class ObjectCopier {
    public static Object copy(Object orig) {
        Class<?> cls = orig.getClass();
        Constructor<?> con = cls.getDeclaredConstructor(cls);

        return ((con == null) ? null : con.newInstance(orig);
    }
}

(未经测试,因此请对待它)