我修改了下面字节码的第15行,并将其从 invokevirtual 更改为 invokespecial(JAVA 8)。不幸的是,我得到一个验证错误(操作数堆栈上的错误类型)
我知道操作数堆栈的值必须是 objectref 中指定的类的子类,但在这种情况下,#18 是 Type 而不是 Type$ClassType,如错误所示。或者换个说法,第15行的堆栈映射框不应该在堆栈中具有Type而不是Type$ClassType吗[0]?我错过了什么?
编辑:堆栈映射帧在更改之前和之后是相同的。(如果我使用的ASM计算帧会改变它们)
Exception Details:
Location:
com/sun/tools/javac/code/Type$ClassType.toString()Ljava/lang/String; @15: invokespecial
Reason:
Type 'com/sun/tools/javac/code/Type' (current frame, stack[0]) is not assignable to 'com/sun/tools/javac/code/Type$ClassType'
Current Frame:
bci: @15
flags: { }
locals: { 'com/sun/tools/javac/code/Type$ClassType', 'java/lang/StringBuilder' }
stack: { 'com/sun/tools/javac/code/Type', 'com/sun/tools/javac/code/TypeTag' }
...
Stackmap Table:
append_frame(@71,Object[#108])
same_frame(@85)
same_frame(@121)
下面是代码。Type$ClassType是Type的直接子类,com/sun/tools/javac/code/Type$ClassType是当前的类,它允许我们使用exactionspecial调用超类(如Type)。
public class com.sun.tools.javac.code.Type$ClassType extends com.sun.tools.javac.code.Type implements
javax.lang.model.type.DeclaredType
....
public java.lang.String toString();
descriptor: ()Ljava/lang/String;
flags: ACC_PUBLIC
Code:
stack=4, locals=2, args_size=1
0: new #108 // class java/lang/StringBuilder
3: dup
4: invokespecial #17 // Method java/lang/StringBuilder."<init>":()V
7: astore_1
8: aload_0
9: invokevirtual #13 // Method com/sun/tools/javac/code/Type$ClassType.getEnclosingType:()Lcom/sun/tools/javac/code/Type;
12: getstatic #10 // Field com/sun/tools/javac/code/TypeTag.CLASS:Lcom/sun/tools/javac/code/TypeTag;
15: invokespecial #18 // Method com/sun/tools/javac/code/Type.hasTag:(Lcom/sun/tools/javac/code/TypeTag;)Z
18: ifeq 71
.....
米琪卡哇伊
湖上湖
相关分类