导致此错误的原因是:java.lang.VerifyError:堆栈高度不一致2!= 1?

时间:2013-11-18 01:21:04

标签: java java-bytecode-asm

我使用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;
    }

2 个答案:

答案 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的两条路径上存在不一致的堆栈深度,导致您看到验证错误。

在我看来,您的错误是L0L2中的块中无用的'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分发的文件。