@GaMen escreveu:
Seguinte, preciso de uma classe que compile e execute código Java, diretamente na memória, sem gerar nenhum arquivo .java ou .class a partir de uma string que contenha o código.
Depois de pesquisar na internet, vi vários exemplos de utilização de classes do pacote
javax.tools
para este fim. Mas quase que a maioria não funcionava, e os que funcionavam não atendiam completamente o que eu precisava.Nestas pesquisas, encontrei um exemplo simples, mas que supostamente atendia o que eu precisava. Este exemplo como a grande maioria, gerava muitos erros quando eu o executava, mas aos poucos fui conseguindo resolver a maioria deles. Segue meu código modificado abaixo.
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 { System.setProperty("java.home", "C:/Program Files/Java/jdk1.8.0_102"); JavaCompiler compiler = ToolProvider.getSystemJavaCompiler(); System.setProperty("java.home", "C:/Program Files/Java/jre1.8.0_102"); 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; } }
O código original pode ser encontrado aqui
Meu problema é que sempre que eu rodo esse código, ao invés de receber a seguinte saída
This is in another java file
no console, eu recebo
No such method: java.lang.NoSuchMethodException: HelloWorld.main([Ljava.lang.String;)
Alguém pode me explicar o porquê disso estar acontecendo, ou me dar uma luz com algum código que faça esse tipo de compilação?
Mensagens: 1
Participantes: 1