Можно ли программно скомпилировать исходный код Java только в памяти? - PullRequest
28 голосов
/ 03 ноября 2011

Я нашел много ссылок, объясняющих, как программно компилировать класс Java с использованием класса JavaCompiler:

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 ]

40 голосов
/ 03 ноября 2011

Для начала посмотрите на 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<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;
  }
}
7 голосов
/ 03 ноября 2011

JavaDocs ваш друг:

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

Посмотрите на последний раздел, который относится к SimpleJavaFileObject; он показывает вам, как использовать его вместе с кодом, который хранится в String

0 голосов
/ 08 октября 2016

Мы рассказали об этом сценарии использования в JavaOne 2016 (вопрос довольно старый, но, похоже, интерес еще есть).

Существует репозиторий с примерами практической генерации кода с использованием javac в памяти.

В частности, посмотрите на SimpleJavaCompiler для примера того, как сделать это в памяти, которая имеет дело с безопасностью потока (мы используем его в контексте сервера) для одного класса. Его можно легко адаптировать для сценария с несколькими классами.

Существуют также классы для загрузки классов и генерации кода (определение переменных, генерация уникальных имен, теневое копирование имен и т. Д.).

...