如何以编程方式编译和实例化Java类?

我将类名称存储在属性文件中。我知道类存储将实现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类?

回答:

如何加载未编译的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();

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

话虽如此,但与实际问题无关:

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

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

properties.load(Thread.currentThread().getContextClassLoader().getResourceAsStream("ClassName.properties"));

以上是 如何以编程方式编译和实例化Java类? 的全部内容, 来源链接: utcz.com/qa/409644.html

回到顶部