JavaCompilerAPI的选项

JavaCompilerAPI的选项,java,compiler-construction,javac,openjdk,Java,Compiler Construction,Javac,Openjdk,我有一个这样的程序:它在内存中编译文件,然后从内存中执行它们。因此,我需要一个带有文件管理器的自定义类加载器,该文件管理器将文件存储在memo://。 现在我想把参数传递给编译器输出的类,因为我认为它不会继承它们(-Xmx80M,-Djava.library.path等等)。我想我需要使用-J选项,但是编译器返回一个IllegalArgumentExceptioncom.sun.tools.javac.main.RecognizedOptions.getjavactoolooptions(nul

我有一个这样的程序:它在内存中编译文件,然后从内存中执行它们。因此,我需要一个带有文件管理器的自定义类加载器,该文件管理器将文件存储在memo://。 现在我想把参数传递给编译器输出的类,因为我认为它不会继承它们(
-Xmx80M
-Djava.library.path
等等)。我想我需要使用-J选项,但是编译器返回一个
IllegalArgumentException
com.sun.tools.javac.main.RecognizedOptions.getjavactoolooptions(null)
也没有列出-J,所以我认为我试图将参数放在错误的位置。我应该在哪里使用-J(或其他选项)有经验吗

编辑:
com.sun.tools.javac.main.RecognizedOptions.getAll(null)
作为选项报告-J,但是
getJavacToolOptions(null
)没有,而且
getJavacFileManagerOptions(null)
也没有

为了澄清,我想将LWJGL库与(运行时)编译代码一起使用。LWJGL需要来自为项目设置的
-Djava.library.path
的一些本机库。但是,编译后的代码无法找到此库路径。我认为它不会继承这个库路径,因此LWJGL抛出一个
NoClassDefFoundError
。否则,它可能会将相对库路径解释为错误memo://lib/lwjgl,但我没有办法检查

堆栈:

Aug 22, 2011 2:14:58 PM customcompile.CustomCompile$2 run
SEVERE: null
java.lang.reflect.InvocationTargetException
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
    at java.lang.reflect.Method.invoke(Method.java:597)
    at customcompile.CustomCompile$2.run(CustomCompile.java:90)
    at java.lang.Thread.run(Thread.java:662)
Caused by: java.lang.NoClassDefFoundError: Could not initialize class org.lwjgl.Sys
    at org.lwjgl.opengl.Display.<clinit>(Display.java:111)
...

如果运行编译器API,编译器将与主程序在同一个VM中运行,并与其共享内存

您正在编译的类没有任何这样的设置——这些是Java VM的设置,而不是任何类的设置。 如果您想稍后在主程序中加载这些类,它们也将与主程序共享内存


因此,我认为使用这些参数没有任何意义。

如果运行编译器API,编译器将与主程序在同一个VM中运行,并与其共享内存

您正在编译的类没有任何这样的设置——这些是Java VM的设置,而不是任何类的设置。 如果您想稍后在主程序中加载这些类,它们也将与主程序共享内存



因此,我认为使用这些参数没有任何意义。

您知道该类使用哪个文件(dll)吗?如果是,请在编译之前手动调用
System.loadLibrary
来解决此问题。

您知道类使用哪个文件(dll)吗?如果您这样做了,请尝试在编译之前手动调用
System.loadLibrary
来解决这个问题。

这是我最初的想法。但是-Djava.library.path在projects-VM参数中设置正确,编译后的代码仍然不会从该路径加载库。如果我的措辞不正确,很抱歉。这也应该有效,但实际上与您原来问题中的问题无关。但不幸的是,它不起作用。我在没有运行时编译器的情况下尝试了完全相同的代码,然后它就完美地工作了。我看了很长时间,在这里找到了-J选项:对不起,你没有真正描述你的问题。你在做什么,你期望发生什么,以及会发生什么?(“不起作用”不是很好的描述。)请将它添加到问题中,而不是注释。它不依赖于编译器,而是依赖于类加载过程(如果有的话)。(但NoClassDefFoundError似乎并不表示缺少库-这将是一个不令人满意的链接错误。请发布错误的整个堆栈跟踪,以及自定义类装入器的相关代码)。这是我最初的想法。但是-Djava.library.path在projects-VM参数中设置正确,编译后的代码仍然不会从该路径加载库。如果我的措辞不正确,很抱歉。这也应该有效,但实际上与您原来问题中的问题无关。但不幸的是,它不起作用。我在没有运行时编译器的情况下尝试了完全相同的代码,然后它就完美地工作了。我看了很长时间,在这里找到了-J选项:对不起,你没有真正描述你的问题。你在做什么,你期望发生什么,以及会发生什么?(“不起作用”不是很好的描述。)请将它添加到问题中,而不是注释。它不依赖于编译器,而是依赖于类加载过程(如果有的话)。(但NoClassDefFoundError似乎并不表示缺少库-这将是一个不满意的链接错误。请发布错误的整个堆栈跟踪,以及自定义类加载器的相关代码)。是否也从自定义类加载器加载OpenGL库?如果是这样,您可以尝试让它们从您的主类加载器加载(并将此加载器指示为您的主类加载器的父级)。您是否也从自定义类加载器加载OpenGL库?如果是这样的话,您可以尝试让它们从主类加载器加载(并将此加载器指示为主类加载器的父级)。我已经对此进行了研究,但是LWJGL具有依赖于操作系统的库,因此我更希望由LWJGL处理此问题,这也是在将来的更新中。@Kaj Toet,只需调用
org.LWJGL.Sys.initialize()
然后。这样更简单。哇,谢谢你的评论!这些库很难加载,因为它们是32位库。我使用的是一个旧的LWJGL,它没有包含64位库。@J-16:您能在答案中添加此注释吗?现在我得到了编译代码中找不到的包,我可以在“原始”代码中完全访问它。我能想到的唯一一件事是我真正改变了,那就是我自己签署了我所有的JAR。我已经研究过了,但是LWJGL有依赖于操作系统的库,因此我更愿意让LWJGL来处理这个问题,也是在将来的更新中。@Kaj Toet,只是cal
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import javax.tools.Diagnostic;
import javax.tools.DiagnosticCollector;
import javax.tools.JavaCompiler;
import javax.tools.JavaFileObject;
import javax.tools.JavaFileObject.Kind;

/**
 *
 * @author Kaj Toet
 */
class MemoryClassLoader extends ClassLoader {


    private JavaCompiler compiler;
    private final MemoryFileManager manager;

    public MemoryClassLoader(JavaCompiler compiler, String classname, String filecontent) {
        this(compiler, Collections.singletonMap(classname, filecontent));
    }

    public MemoryClassLoader(JavaCompiler compiler, Map<String, String> map) {
            this.compiler=compiler;

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

            manager  = new MemoryFileManager(this.compiler);
            List<Source> list = new ArrayList<Source>();
            for (Map.Entry<String, String> entry : map.entrySet()) {
                list.add(new Source(entry.getKey(), Kind.SOURCE, entry.getValue()));
            }            

            List<String> optionList = new ArrayList<String>();
            // set compiler's classpath to be same as the runtime's
            //optionList.addAll(Arrays.asList("-cp", ".."));

            this.compiler.getTask(null, this.manager, diagnostics, optionList, null, list).call();
            for (Diagnostic diagnostic : diagnostics.getDiagnostics()) {
              CustomCompile.addDebugText(diagnostic.toString());
            }
    }
    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        synchronized (this.manager) {
            Output mc = this.manager.map.remove(name);
            if (mc != null) {
                byte[] array = mc.toByteArray();
                return defineClass(name, array, 0, array.length);
            }
        }
        return super.findClass(name);
    }
}
class MemoryFileManager extends ForwardingJavaFileManager<JavaFileManager> {
    public final Map<String, Output> map = new HashMap<String, Output>();

    MemoryFileManager(JavaCompiler compiler) {
        super(compiler.getStandardFileManager(null, null, null));
    }

    @Override
    public Output getJavaFileForOutput
            (Location location, String name, Kind kind, FileObject source) {
        Output mc = new Output(name, kind);
        this.map.put(name, mc);
        return mc;
    }  
}
class Output extends SimpleJavaFileObject {
    private final ByteArrayOutputStream baos = new ByteArrayOutputStream();

    Output(String name, Kind kind) {
        super(URI.create("memo:///" + name.replace('.', '/') + kind.extension), kind);
    }

    byte[] toByteArray() {
        return this.baos.toByteArray();
    }

    @Override
    public ByteArrayOutputStream openOutputStream() {
        return this.baos;
    }
}