在 Java 中从字符串运行代码请不要实际使用这个

2022-09-04 21:11:44

出于调试原因,我希望能够运行通过控制台键入的代码。例如:

BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
while(true){
    String str = br.readLine(); //This can return 'a = 5;','b = "Text";' or 'pckg.example.MyClass.run(5);'
    if(str == null)
        return;
    runCode(str); //How would I do this?
}

答案 1

请不要实际使用这个

我假设你想将字符串作为Java代码来评估,而不是像Javascript这样的脚本引擎,所以我在阅读本文后一时兴起地创建了这个,使用了提到的编译器API标记。这可能是非常糟糕的做法,但它(在某种程度上)像你想要的那样工作。我怀疑它在调试中会有多大用处,因为它在新类的上下文中运行代码。示例用法包含在底部。

import javax.tools.JavaCompiler;
import javax.tools.StandardJavaFileManager;
import javax.tools.ToolProvider;
import java.io.BufferedReader;
import java.io.File;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.Arrays;

public class main {
    public static void runCode(String s) throws Exception{
        JavaCompiler jc = ToolProvider.getSystemJavaCompiler();
        StandardJavaFileManager sjfm = jc.getStandardFileManager(null, null, null);
        File jf = new File("test.java"); //create file in current working directory
        PrintWriter pw = new PrintWriter(jf);
        pw.println("public class test {public static void main(){"+s+"}}");
        pw.close();
        Iterable fO = sjfm.getJavaFileObjects(jf);
        if(!jc.getTask(null,sjfm,null,null,null,fO).call()) { //compile the code
            throw new Exception("compilation failed");
        }
        URL[] urls = new URL[]{new File("").toURI().toURL()}; //use current working directory
        URLClassLoader ucl = new URLClassLoader(urls);
        Object o= ucl.loadClass("test").newInstance();
        o.getClass().getMethod("main").invoke(o);

    }
    public static void main(String[] args) {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        while(true){
            try {
            String str = br.readLine(); //This can return 'a = 5;','b = "Text";' or 'pckg.example.MyClass.run(5);'
            if(str == null)
                return;

            runCode(str); //How would I do this?
            } catch(Exception e) {
                e.printStackTrace();
            }
        }

    }
}
//command line
>  System.out.println("hello");
hello
>  System.out.println(3+2+3+4+5+2);
19
>  for(int i = 0; i < 10; i++) {System.out.println(i);}
0
1
2
3
4
5
6
7
8
9

使用SimpleJavaFileObject,您实际上可以避免使用文件,如下所示,但是语法似乎有点麻烦,所以我只是在当前工作目录中选择了一个文件。

编辑将字符串转换为代码提供了类似的方法,但它没有完全充实


答案 2

如果代码是JavaScript,那么你可以用JavaScript引擎运行它:

Object res = new ScriptEngineManager().getEngineByName("js").eval(str);

JavaScript 引擎自 1.6 起就是 Java SE 的一部分。有关详细信息,请参阅本指南 http://download.java.net/jdk8/docs/technotes/guides/scripting/programmer_guide/index.html


推荐