72

我将类名存储在属性文件中。我知道类存储将实现 IDynamicLoad。如何动态实例化类?

现在我有

     Properties foo = new Properties();
    foo.load(new FileInputStream(new File("ClassName.properties")));
    String class_name = foo.getProperty("class","DefaultClass");
    //IDynamicLoad newClass = Class.forName(class_name).newInstance();

newInstance 是否仅加载已编译的 .class 文件?如何加载未编译的 Java 类?

4

3 回答 3

146

如何加载未编译的 Java 类?

你需要先编译它。这可以通过javax.toolsAPI以编程方式完成。这只需要在JRE 之上的本地机器上安装JDK 。

这是一个基本的启动示例(抛开明显的异常处理):

// Prepare source somehow.
String source = "package test; public class Test { static { System.out.println(\"hello\"); } public Test() { System.out.println(\"world\"); } }";

// Save source in .java file.
File root = new File("/java"); // On Windows running on C:\, this is C:\java.
File sourceFile = new File(root, "test/Test.java");
sourceFile.getParentFile().mkdirs();
Files.write(sourceFile.toPath(), source.getBytes(StandardCharsets.UTF_8));

// Compile source file.
JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
compiler.run(null, null, null, sourceFile.getPath());

// Load and instantiate compiled class.
URLClassLoader classLoader = URLClassLoader.newInstance(new URL[] { root.toURI().toURL() });
Class<?> cls = Class.forName("test.Test", true, classLoader); // Should print "hello".
Object instance = cls.newInstance(); // Should print "world".
System.out.println(instance); // Should print "test.Test@hashcode".

哪个产量像

hello
world
test.Test@ab853b

implements如果这些类的某个接口已经在类路径中,则进一步使用会更容易。

SomeInterface instance = (SomeInterface) cls.newInstance();

否则,您需要使用反射 API来访问和调用(未知)方法/字段。


也就是说,与实际问题无关:

properties.load(new FileInputStream(new File("ClassName.properties")));

java.io.File依赖于当前工作目录是可移植性问题的秘诀。不要那样做。将该文件放在类路径中并ClassLoader#getResourceAsStream() 与类路径相对路径一起使用。

properties.load(Thread.currentThread().getContextClassLoader().getResourceAsStream("ClassName.properties"));
于 2010-05-31T23:11:03.390 回答
6

与 BalusC 的答案相同,但在我的 kilim 发行版的这段代码中,这里有一个更自动的包装器。 https://github.com/kilim/kilim/blob/master/src/kilim/tools/Javac.java

它获取包含 Java 源代码的字符串列表,提取包和公共类/接口名称,并在 tmp 目录中创建相应的目录/文件层次结构。然后它在其上运行 java 编译器,并返回名称、类文件对(ClassInfo 结构)的列表。

帮助自己编写代码。它是麻省理工学院许可的。

于 2011-10-07T03:14:17.850 回答
5

如果您知道该类具有公共无参数构造函数,则您的注释代码是正确的。您只需要转换结果,因为编译器无法知道该类实际上会实现IDynamicLoad. 所以:

   IDynamicLoad newClass = (IDynamicLoad) Class.forName(class_name).newInstance();

当然,这个类必须被编译并在类路径上才能工作。

如果您希望从源代码动态编译一个类,那就是另一回事了。

于 2010-05-31T23:03:33.067 回答