是否有可能以编程方式仅在内存中编译java源代码?

时间:2011-11-03 00:53:23

标签: java code-generation java-compiler-api

我找到了许多引用,解释了如何使用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);

感谢您的帮助。

3 个答案:

答案 0 :(得分:39)

首先,请查看JavaCompiler API。基本上是:

  1. 以字符串形式创建Java类。
  2. 将字符串放入扩展SimpleJavaFileObject
  3. 的类中
  4. 使用JavaCompiler实例进行编译。
  5. 最后,将方法称为新类。


    这是一个与JDK6 +一起使用的example

    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<JavaFileObject> diagnostics = new DiagnosticCollector<JavaFileObject>();
    
        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;
      }
    }
    

答案 1 :(得分:7)

JavaDocs是你的朋友:

http://download.oracle.com/javase/6/docs/api/javax/tools/JavaCompiler.html

查看引用SimpleJavaFileObject的最后一节;它向您展示了如何将其与存储在String

中的代码结合使用

答案 2 :(得分:0)

我们在JavaOne 2016上讨论了这个用例(问题有点陈旧,但似乎仍有一些兴趣)。

有一个repository,其中包含使用javac in-memory生成实际代码的示例。

具体看看SimpleJavaCompiler有关如何在内存中处理单个类的线程安全(我们在服务器的上下文中使用它)的示例。它可以很容易地适应多类场景。

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