如何加载未编译的 Java 类?
你需要先编译它。这可以通过javax.tools
API以编程方式完成。这只需要在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"));