gpt4 book ai didi

compiler-construction - JavaCompiler API 选项

转载 作者:行者123 更新时间:2023-11-29 08:12:43 28 4
gpt4 key购买 nike

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

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

澄清一下,我想将 LWJGL 库与(运行时)编译代码一起使用。 LWJGL 需要为项目设置的 -Djava.library.path 中的一些 native 库。但是编译后的代码无法找到这个库路径。我认为它不会继承此库路径,因此 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)
...

我还必须注意项目包含的库已成功加载,但是 LWJGL 库加载了额外的 native 库 - 我认为这不起作用。

自定义类加载器: 包自定义编译;

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;
}
}

最佳答案

如果您运行编译器 API,编译器将与您的主程序在同一虚拟机中运行,并与其共享内存。

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

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

关于compiler-construction - JavaCompiler API 选项,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7141774/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com