是否有可能以编程方式仅在内存中编译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 。 基本上:

  1. 在字符串中创建Java类。
  2. 将字符串放入扩展SimpleJavaFileObject的类中。
  3. 使用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 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 ,了解如何在内存中处理单个类的线程安全(我们在服务器的上下文中使用它)。 它可以很容易地适应多类场景。

还有一些类来处理类加载和代码生成(变量的范围,生成唯一的名称,名称阴影等)。