我想创建一个生成Java字节码的简单编译器。为了生成代码,我正在使用Apache BCEL库。
但是,我创建了一个简单的类,但没有做任何事情的主方法。我使用下面的代码(我知道它是Scala,但这不是很重要,你将在后面看到)生成类文件:
private def generateClassFile(): Unit =
{
// JVM tutorial: https://commons.apache.org/proper/commons-bcel/manual/jvm.html
val interfaces: Array[String] = new Array(0);
classFactory = new ClassGen(
"MiniPascal",
"java.lang.Object",
null,
Const.ACC_PUBLIC | Const.ACC_SUPER,
interfaces
);
val mainConstantPool: ConstantPoolGen = new ConstantPoolGen();
val mainMethod = generateMainMethod(mainConstantPool);
classFactory.addMethod(mainMethod.getMethod());
}
private def generateMainMethod(mainConstantPool): MethodGen =
{
val instructions = new InstructionList();
instructions.append(InstructionConstants.NOP);
mainConstantPool.addNameAndType("main", "([java/lang/String;)V");
val methodArgumentNames = Array("args");
val methodArgumentTypes: Array[Type] = Array(new ArrayType(Type.STRING, 1));
val mainMethod: MethodGen = new MethodGen(
Const.ACC_PUBLIC | Const.ACC_STATIC,
Type.VOID,
methodArgumentTypes,
methodArgumentNames,
"main", "MiniPascal",
instructions,
mainConstantPool
);
return mainMethod;
}
private def saveClassFile(): Unit =
{
val classFile: JavaClass = classFactory.getJavaClass();
classFile.dump("MiniPascal.class");
//println(classFile.toString());
}
运行程序时会生成类文件,但是在执行java MiniPascal
时会出现以下异常:
java.lang.ClassFormatError: Illegal constant pool index 4 for method signature in class file MiniPascal
我读到了Java字节码并知道常量池大约是类文件内容的60%并存储了所有文字的名称,但我还没有弄清楚异常的情况。
我期待着您的建议。谢谢你的帮助!
答案 0 :(得分:2)
您正在使用line:
创建自己不与该类关联的空常量池val mainConstantPool: ConstantPoolGen = new ConstantPoolGen();
因此,虽然您将方法签名添加到常量池中,但它不起作用,因为常量池从未添加到类文件中。
首先创建常量池并将其作为最后一个参数传递给ClassGen
构造函数,或者从mainConstantPool
初始化classFactory.getConstantPool()
。