我如何以编程方式编译和实例化Java类?
我有存储在属性文件中的类名称。 我知道类存储将实现IDynamicLoad。 我如何dynamic实例化类?
现在我有
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.tools
API以编程方式完成。 这只需要将JDK安装在JRE之上的本地计算机上。
下面是一个基本的启动示例(抛开明显的exception处理):
// 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
了已经在类path中的某个接口,则进一步的使用会更容易。
SomeInterface instance = (SomeInterface) cls.newInstance();
否则,您需要使用Reflection API来访问和调用(未知)方法/字段。
这与实际问题无关:
properties.load(new FileInputStream(new File("ClassName.properties")));
让java.io.File
依赖于当前的工作目录是可移植性问题的秘诀。 不要这样做。 将该文件放入classpath中,并使用ClassLoader#getResourceAsStream()
path相对path使用ClassLoader#getResourceAsStream()
。
properties.load(Thread.currentThread().getContextClassLoader().getResourceAsStream("ClassName.properties"));
与BalusC的答案一样,但是在我的kilim发行版的这段代码中有一点更多的自动包装。 https://github.com/kilim/kilim/blob/master/src/kilim/tools/Javac.java
它包含一个包含Java源代码的string列表,提取包和公共类/接口名称,并在tmp目录中创build相应的目录/文件层次结构。 然后运行它上面的java编译器,并返回名称,类文件对(ClassInfo结构)的列表。
帮助自己的代码。 这是MIT许可的。
如果您知道该类具有公共无参数构造函数,则您的注释代码是正确的。 您只需要输出结果,因为编译器无法知道该类实际上会实现IDynamicLoad
。 所以:
IDynamicLoad newClass = (IDynamicLoad) Class.forName(class_name).newInstance();
当然,这个类必须被编译,并在类path上工作。
如果你正在寻找从源代码dynamic编译一个类,这是一个完整的其他水壶。