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

时间:2011-05-23 08:20:35

标签: java reflection instantiation

是否有办法在给定类名(动态)的情况下创建特定类的实例,并将参数传递给其构造函数。

类似的东西:

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

"MyAttributeValue"MyClass的构造函数的参数。

10 个答案:

答案 0 :(得分:464)

是的,比如:

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")

答案 1 :(得分:89)

您可以使用Class.forName()来获取所需类别的Class对象。

然后使用getConstructor()查找所需的Constructor对象。

最后,在该对象上调用newInstance()以获取新实例。

Class<?> c = Class.forName("mypackage.MyClass");
Constructor<?> cons = c.getConstructor(String.class);
Object object = cons.newInstance("MyAttributeValue");

答案 2 :(得分:78)

您可以使用反射

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

答案 3 :(得分:12)

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

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

答案 4 :(得分:8)

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

如果您想访问非公开构造函数,则必须使用(例如)getDeclaredConstructor(String.lang)

答案 5 :(得分:4)

答案 6 :(得分: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" }); 

这里Base类应该看起来像:

public class Base {

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

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

答案 7 :(得分: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);

答案 8 :(得分:1)

如果有人正在寻找一种创建类实例的方法,尽管该类遵循Singleton模式,这是一种方法。

// 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();

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

答案 9 :(得分:0)

另一个有用的答案。 How do I use 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);