是否有可能以编程方式仅在内存中编译java源代码?
我找到了很多参考资料,解释了如何使用JavaCompiler
类以编程方式编译Java类:
JavaCompiler compiler = ToolProvider.getSystemJavaCompiler(); int result = compiler.run(null, null, null, "a_file_name");
但是,我想知道是否有一个开源库让我编译以编程方式生成的源代码(因此不涉及src文件)并在输出流中生成一些字节代码(不在文件系统中生成类文件) )。
例如,我正在寻找能够写这样的东西:
InputStream input = generateSourceCode(); OutputStream output = getByteCode(input); doCoolStuffWithByteCode(output);
谢谢你的帮助。
首先,查看JavaCompiler API 。 基本上:
- 在字符串中创建Java类。
- 将字符串放入扩展SimpleJavaFileObject的类中。
- 使用
JavaCompiler
实例进行编译。
最后,将方法称为新类。
这是一个适用于JDK6 +的示例 :
import java.io.IOException; import java.io.PrintWriter; import java.io.StringWriter; import java.lang.reflect.InvocationTargetException; import java.net.URI; import java.util.Arrays; import javax.tools.Diagnostic; import javax.tools.DiagnosticCollector; import javax.tools.JavaCompiler; import javax.tools.JavaFileObject; import javax.tools.SimpleJavaFileObject; import javax.tools.ToolProvider; import javax.tools.JavaCompiler.CompilationTask; import javax.tools.JavaFileObject.Kind; public class CompileSourceInMemory { public static void main(String args[]) throws IOException { JavaCompiler compiler = ToolProvider.getSystemJavaCompiler(); DiagnosticCollector diagnostics = new DiagnosticCollector (); StringWriter writer = new StringWriter(); PrintWriter out = new PrintWriter(writer); out.println("public class HelloWorld {"); out.println(" public static void main(String args[]) {"); out.println(" System.out.println(\"This is in another java file\");"); out.println(" }"); out.println("}"); out.close(); JavaFileObject file = new JavaSourceFromString("HelloWorld", writer.toString()); Iterable extends JavaFileObject> compilationUnits = Arrays.asList(file); CompilationTask task = compiler.getTask(null, null, diagnostics, null, null, compilationUnits); boolean success = task.call(); for (Diagnostic diagnostic : diagnostics.getDiagnostics()) { System.out.println(diagnostic.getCode()); System.out.println(diagnostic.getKind()); System.out.println(diagnostic.getPosition()); System.out.println(diagnostic.getStartPosition()); System.out.println(diagnostic.getEndPosition()); System.out.println(diagnostic.getSource()); System.out.println(diagnostic.getMessage(null)); } System.out.println("Success: " + success); if (success) { try { Class.forName("HelloWorld").getDeclaredMethod("main", new Class[] { String[].class }) .invoke(null, new Object[] { null }); } catch (ClassNotFoundException e) { System.err.println("Class not found: " + e); } catch (NoSuchMethodException e) { System.err.println("No such method: " + e); } catch (IllegalAccessException e) { System.err.println("Illegal access: " + e); } catch (InvocationTargetException e) { System.err.println("Invocation target: " + e); } } } } class JavaSourceFromString extends SimpleJavaFileObject { final String code; JavaSourceFromString(String name, String code) { super(URI.create("string:///" + name.replace('.','/') + Kind.SOURCE.extension),Kind.SOURCE); this.code = code; } @Override public CharSequence getCharContent(boolean ignoreEncodingErrors) { return code; } }
JavaDocs是你的朋友:
http://download.oracle.com/javase/6/docs/api/javax/tools/JavaCompiler.html
查看引用SimpleJavaFileObject
的最后一节; 它向您展示了如何将它与存储在String
代码结合使用
我们在JavaOne 2016中讨论了这个用例(问题有点陈旧,但似乎仍有一些兴趣)。
有一个存储库 ,其中包含使用内存中的javac实际代码生成的示例。
具体来说,请看SimpleJavaCompiler ,了解如何在内存中处理单个类的线程安全(我们在服务器的上下文中使用它)。 它可以很容易地适应多类场景。
还有一些类来处理类加载和代码生成(变量的范围,生成唯一的名称,名称阴影等)。