使用类名称创建实例并调用构造函数


312

有没有一种方法可以给定类名称(动态)来创建特定类的实例,并将参数传递给其构造函数。

就像是:

Object object = createInstance("mypackage.MyClass","MyAttributeValue");

"MyAttributeValue"的构造函数的参数在哪里MyClass

Answers:


497

是的,类似:

Class<?> clazz = Class.forName(className);
Constructor<?> ctor = clazz.getConstructor(String.class);
Object object = ctor.newInstance(new Object[] { ctorArgument });

当然,这仅适用于单个字符串参数,但是您可以轻松地对其进行修改。

注意,类名必须是完全合格的,即包括名称空间。对于嵌套类,您需要使用美元(因为这是编译器使用的美元)。例如:

package foo;

public class Outer
{
    public static class Nested {}
}

要获得该Class对象,您需要Class.forName("foo.Outer$Nested")


5
newInstance()是varargs方法(与一样GetConstructor()),无需显式Object创建-array。
约阿希姆·绍尔

18
@Joachim:我知道它是varargs,但是当您有一个Object[]参数时它可能会变得棘手,因此我更喜欢在这种情况下显式创建数组。
乔恩·斯基特

2
clazz.getConstructor(String.class); 为什么在这里使用String.class?
Umair A.

2
@Neutralizer:是的,但是我在回答一个不需要动态的问题。
乔恩·斯基特

3
@JonSkeet我知道您来自哪里,但是并不是那么简单-我确实看过文档但是很困惑,但是如果我对其进行了测试并且可以正常工作-可以,但是可以正常工作-但是如果不正常,那么我不确定问题是否是由于配置不足或我个人的某些原因所致-通常在问这样简单的问题时,人们会抛出有用的小窍门,这确实有帮助。这就是为什么简单的“是可行的-如果您以这种方式这样做”或“没有没有办法”确实有帮助的原因。但是我现在的理解是没有办法
ycomp


81

您可以使用反射

return Class.forName(className).getConstructor(String.class).newInstance(arg);

3
如果使用默认构造函数,则删除String.class参数值,例如return Class.forName(className).getConstructor()。newInstance(arg);。
2013年

21
@VijayKumar我想你是说Class.forName(className).getConstructor().newInstance();;)
彼得·劳瑞

14

如果class只有一个空的构造函数(例如Activity或Fragment等,android类):

Class<?> myClass = Class.forName("com.example.MyClass");    
Constructor<?> constructor = myClass.getConstructors()[0];

2
这就是帮助我的原因。Constructor<?> ctor = clazz.getConstructor(String.class)似乎不适合我。
Leo C Han

8

使用(即)getConstructor(String.lang)构造函数时,必须将其声明为public。否则NoSuchMethodException抛出a。

如果要访问非公共构造函数,则必须改用(ie)getDeclaredConstructor(String.lang)



4

通过Class<?>构造函数自变量使用Java创建对象的非常简单的方法:

情况1:- 这是此类中的一小段代码Main

import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;

public class Main {

    public static void main(String args[]) throws ClassNotFoundException, NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {

        // Get class name as string.
        String myClassName = Base.class.getName();
        // Create class of type Base.
        Class<?> myClass = Class.forName(myClassName);
        // Create constructor call with argument types.
        Constructor<?> ctr = myClass.getConstructor(String.class);
        // Finally create object of type Base and pass data to constructor.
        String arg1 = "My User Data";
        Object object = ctr.newInstance(new Object[] { arg1 });
        // Type-cast and access the data from class Base.
        Base base = (Base)object;
        System.out.println(base.data);
    }

}

并且,这是Base类结构:

public class Base {

    public String data = null;

    public Base() 
    {
        data = "default";
        System.out.println("Base()");
    }

    public Base(String arg1) {
        data = arg1;
        System.out.println("Base("+arg1+")");
    }

}

情况2:您可以为具有多个参数的构造函数类似地编写代码,并复制构造函数。例如,将3个参数作为参数传递给Base构造函数将需要在类中创建构造函数,并且上面的代码更改如下:

Constructor<?> ctr = myClass.getConstructor(String.class, String.class, String.class);
Object object = ctr.newInstance(new Object[] { "Arg1", "Arg2", "Arg3" }); 

在这里,基类应该看起来像:

public class Base {

    public Base(String a, String b, String c){
        // This constructor need to be created in this case.
    }   
}

注意:- 不要忘记处理代码中需要处理的各种异常。


另一种方法也是通过clone()现有的java对象。这将创建一个现有Java对象的副本。对于这种情况,您还必须处理深层副本或浅层副本的概念。
Rahul Raina

2

如果尽管遵循Singleton Pattern的类,但有人在寻找一种创建类实例的方法,则可以采用这种方法。

// Get Class instance
Class<?> clazz = Class.forName("myPackage.MyClass");

// Get the private constructor.
Constructor<?> cons = clazz.getDeclaredConstructor();

// Since it is private, make it accessible.
cons.setAccessible(true);

// Create new object. 
Object obj = cons.newInstance();

这仅适用于使用私有构造函数实现单例模式的类。


1

您也可以在创建的对象内调用方法。

您可以通过调用第一个承包商来立即创建对象,然后在创建的对象中调用第一个方法。

    Class<?> c = Class.forName("mypackage.MyClass");
    Constructor<?> ctor = c.getConstructors()[0];
    Object object=ctor.newInstance(new Object[]{"ContstractorArgs"});
    c.getDeclaredMethods()[0].invoke(object,Object... MethodArgs);

您怎么知道第一个构造函数将其String作为参数?更改构造函数顺序时会变得有点凌乱
Farid

@来自班级文档的内容
Hatem Badawi

getConstructor(ClassName.class)我猜还是更好。即使构造函数的顺序在类中发生变化,也无需手动查找位置
Farid

@Farid-c.getDeclaredMethods()[0] .invoke(object,Object ... MethodArgs); 在某些情况下指定特殊的构造函数,您可能需要这样做;但是您是对的。
哈特姆·巴达维

1

另一个有用的答案。如何使用getConstructor(params).newInstance(args)?

return Class.forName(**complete classname**)
    .getConstructor(**here pass parameters passed in constructor**)
    .newInstance(**here pass arguments**);

就我而言,我的类的构造函数将Webdriver作为参数,因此在以下代码中使用:

return Class.forName("com.page.BillablePage")
    .getConstructor(WebDriver.class)
    .newInstance(this.driver);
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.