使用反射无法正常工作的Java运行时编译器

时间:2017-02-01 17:35:14

标签: java class reflection runtime javacompiler

我有一个JavaFX应用程序,其中有一个编辑器。在编辑器中,用户将能够编写java代码,并且我有一个按钮来编译此代码并运行main方法。例如,编辑器将包含以下代码:

public class Test {
    public static void main(String[] args) {
        System.out.println("hello");
    }
}

点击按钮,将运行以下代码:

runButton.setOnAction(e -> {
        compiler.set(editor.getText());
        try {
            compiler.createFile();
        } catch (IOException e1) {
            e1.printStackTrace();
        }
        compiler.compile();
        compiler.run();

    });

在编译器类中,有以下实现:

public class CodeCompiler {

public String className;
public String code;

public void set(String code) {
    try {
        this.className = code.substring(code.indexOf(" class ") + 6, code.indexOf(" {")).trim();
    } catch(StringIndexOutOfBoundsException e) {

    }
    this.code = code;
}

public void createFile() throws IOException {
    PrintWriter pw = new PrintWriter("speech2code/src/main/java/" + className + ".java");
    pw.close();
    FileWriter writer = new FileWriter("speech2code/src/main/java/" + className + ".java", true);
    writer.write(code);
    writer.flush();
    writer.close();
}

public void compile() {
    File file = new File("speech2code/src/main/java/" + className + ".java");
    File classFile = new File("speech2code/src/main/java/" + className + ".class");
    classFile.delete(); // delete class file f it exists
    JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
    compiler.run(null, null, null, file.getPath());
}

public void run() {

    Class<?> cls = null;
    try {
        cls = Class.forName(className);
        System.out.println(cls == null);
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    }
    Method meth = null;
    try {
        meth = cls.getMethod("main", String[].class);
    } catch (NoSuchMethodException e) {
        e.printStackTrace();
    }
    String[] params = null;
    try {
        meth.invoke(null, (Object) params);
    } catch (IllegalAccessException e) {
        e.printStackTrace();
    } catch (InvocationTargetException e) {
        e.printStackTrace();
    }

}


}

现在上面的代码成功创建了java文件,类文件并第一次正确运行。现在,当我更改编辑器代码以打印其他内容时,它会输出代码第一次运行时的结果。因此,在这种情况下,它仍然会打印“你好”。而不是它当前的价值。

任何问题都可能出错? 谢谢!

1 个答案:

答案 0 :(得分:0)

您需要为新类创建一个新的类加载器。该类不会因为您编译而重新加载。

 URLClassLoader classLoader = URLClassLoader.newInstance(new URL[] {classFile});

然后你可以问这个装载机上课:

Class<?> cls = Class.forName(className, true, classLoader);