我使用ASM为while()语句生成字节代码。但是eclipse报道:
Exception in thread "main" java.lang.VerifyError: (class: show_cise_image, method: main signature: ([Ljava/lang/String;)V) Inconsistent stack height 2 != 1
at java.lang.Class.getDeclaredMethods0(Native Method)
..................
我的字节码源代码:
show_cise_image {
boolean flag;
flag = true;
while(flag){
flag = false;
}
}
生成上述代码的字节代码:
/ class version 51.0 (51)
// access flags 0x21
public class show_cise_image {
// access flags 0x8
static int v = 0
// access flags 0x8
static boolean flag = 0
// access flags 0x9
public static main(String[]) : void
L0
LINENUMBER 6 L0
GETSTATIC show_cise_image.flag : boolean
LDC 1
PUTSTATIC show_cise_image.flag : boolean
GOTO L1
L2
GETSTATIC show_cise_image.flag : boolean
LDC 0
PUTSTATIC show_cise_image.flag : boolean
L1
GETSTATIC show_cise_image.flag : boolean
IFNE L2
RETURN
L3
LOCALVARIABLE args String[] L0 L3 2
LOCALVARIABLE x int L0 L3 0
LOCALVARIABLE y int L0 L3 1
MAXSTACK = 3
MAXLOCALS = 3
}
我的java代码生成字节码(我认为这个错误是由while()语句引起的,所以我只发布这部分):
/* while(Expr){ stmt*} */
@Override
public Object visitIterationStmt(IterationStmt iterationStmt, Object arg)
throws Exception {
MethodVisitor mv = (MethodVisitor)arg;
Label guardLabel = new Label();
Label bodyLabel = new Label();
mv.visitJumpInsn(GOTO, guardLabel);
mv.visitLabel(bodyLabel);
for(Stmt t : iterationStmt.stmtList)
t.visit(this, mv); // execute statements in body
mv.visitLabel(guardLabel);
iterationStmt.expr.visit(this, mv); // put the result of expr on stack
mv.visitJumpInsn(IFNE, bodyLabel);
return null;
}
答案 0 :(得分:6)
让我们手动分析您的字节码:
L0 ; on entry stack is empty
LINENUMBER 6 L0
GETSTATIC show_cise_image.flag : boolean ; pushes a value, stack height is 1
LDC 1 ; pushes a value, stack heighe is 2
PUTSTATIC show_cise_image.flag : boolean ; pop 1 value, stack height is 1
GOTO L1 ; stack height 1 on going to L1...
L1 ; stack height 1 from previous goto
GETSTATIC show_cise_image.flag : boolean ; pushes a value, stack height is 2
IFNE L2 ; pops 1 value for test, stack height 1 on branch
L2 ; stack height 1 from previous branch
GETSTATIC show_cise_image.flag : boolean ; pushes a value, stack height is 2
LDC 0 ; pushes a value, stack height is 3
PUTSTATIC show_cise_image.flag : boolean ; pops a value, stack height is 2
; fall through to L1 with stack height 2
因此,在L1
的两条路径上存在不一致的堆栈深度,导致您看到验证错误。
在我看来,您的错误是L0
和L2
中的块中无用的'GETSTATIC'字节码 - 您正在推送堆栈上flag
的值,但从来没有做过任何事情。
答案 1 :(得分:0)
这是因为内置的tomcat编译器。
Eclipse JDT中的Java编译器作为默认编译器包含在内。它是一个高级Java编译器,它将从Tomcat类加载器加载所有依赖项,这将在编译具有数十个JAR的大型安装时提供极大的帮助。在快速服务器上,这将允许甚至大型JSP页面的亚秒级重新编译周期。
如果出现任何此类问题,请按照以下解决方法进行操作
在以前的Tomcat版本中使用的Apache Ant,可以通过简单地删除lib / ecj - * .jar文件,并放置ant.jar和ant来代替新编译器的 -launcher.jar来自lib文件夹中最新Ant分发的文件。