1

我不是 Java 专家,而且我对编译和运行动态生成代码的整个概念还很陌生,这在其他语言中非常简单,特别是 Javascript 和 PHP 等脚本语言。

我正在关注这段代码: http ://www.java2s.com/Code/Java/JDK-6/CompilingfromMemory.htm 我做了这样的事情:

private final String = "GeneratedClass_" + Long.toHexString(random.nextLong());
private Method compileCode(String code) {
    JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
    if (compiler == null) return null;

    DiagnosticCollector<JavaFileObject> diagnostics = new DiagnosticCollector<JavaFileObject>();

    JavaFileObject source = new JavaSource(className, code);
    Iterable<? extends JavaFileObject> compilationUnits = Arrays.asList(source);
    CompilationTask task = compiler.getTask(null, null, diagnostics, null, null, compilationUnits);

    if (!task.call()) return null;
    try {
        return Class.forName(className).getDeclaredMethods()[0];
    } catch (ClassNotFoundException e) {}
    return null;
}

private class JavaSource extends SimpleJavaFileObject {
    final String code;
    JavaSource(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;}
}

想象一下,字符串代码类似于

"public class GeneratedClass_65ce701239c32ce0 {
    public String hello() {
        return "Hello, world!";
    }
}"

在抛出 ClassNotFoundException 的 Class.forName 之前它运行良好。我很困惑,因为我似乎没有从片段中删除一些重要的东西:所以,这个类已经编译了,但是它去哪儿了?

我读过一些关于使用不同类加载器的文章,但是就像我说的那样,我对所有这些东西都很陌生,我不知道去哪里以及如何使用它,以及我应该如何定义自己的 ClassLoader 扩展. 我唯一知道的是,一切对我来说似乎都很复杂......

在 Windows 7 和 JDK 1.7 中使用 Eclipse Indigo。

4

1 回答 1

2

你删掉的一件重要的事情是所有的错误输出和诊断信息。你永远不会知道是否出了什么问题。然而,一切看起来都是正确的。您的问题很可能只是您没有向编译器发送任何选项,因此它会将类文件写入任何感觉的位置(我相信当前工作目录是默认目录),这可能不在您的类路径,尤其是在 IDE 中。尝试从命令行运行它以向自己证明它有效。这应该有效:

mkdir tmp
javac -d tmp <path your main class .java file>
java -cp .;tmp <your main class name>

如果您不熟悉命令行工具,则 to 的参数必须javac是 .java 文件的文件系统路径,而 to 的参数java必须是 . 分隔的全限定类名,例如com.foo.Main. 这样做应该:

  1. 将您的类编译到 tmp 目录。
  2. 将动态生成的类写入当前目录。
  3. 从当前目录成功加载新编译的类,因为它在类路径上。
于 2011-10-16T00:36:38.790 回答