Invokespecial 验证错误:类型不可分配

我修改了下面字节码的第 15 行,并将其从 invokevirtual 更改为 invokespecial (JAVA 8)。不幸的是,我收到了一个验证错误(操作数堆栈上的类型错误)


我知道操作数堆栈的值必须是 objectref 中指定的类的子类,但在这种情况下,#18 是 Type 而不是 Type$ClassType ,就像错误提示的那样。或者换一种说法,第15行的stackmapframe不应该在stack[0]中有Type而不是Type$ClassType吗?我错过了什么?


编辑:stackmapframes 在更改之前和之后是相同的。(如果我使用的 ASM COMPUTE FRAMES 会改变它们)


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 是当前类,它允许我们使用 invokespecial 调用超类(如 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



倚天杖
浏览 141回答 2
2回答

绝地无双

invokespecial用于实现三件事之一构造函数调用调用private方法super. …打电话_虽然 1. 在这里不适用(因为目标方法的名称不是<init>),但其他任何一种情况都要求接收器类型是当前类或其子类。因此,即使方法的声明类是Type,实际接收者的类型也应该可以分配给当前类,Type$ClassType。与您通过更改创建的最接近的等价物是super调用,尽管在 Java 源代码中,调用方法 viasuper强制接收器引用与 相同this,它本质上可分配给当前类。在字节码级别,规则限制较少,但不允许在可能指向完全不相关的子类层次结构的实例的类型引用上调用允许绕过当前类或其子类中的方法声明的方法调用,即Type不是一个Type$ClassType。apangin 的回答中已经引用了相关的 JVMS 规则。

肥皂起泡泡

您尝试在(@9 返回)invokespecial的实例上执行,而验证器期望当前类的引用,即.TypeinvokevirtualType$ClassType请参阅 JVMS&nbsp;§4.10.1.9:可以将传入操作数堆栈上与当前类和 Descriptor 中给出的参数类型匹配的类型有效地替换为 Descriptor 中给出的返回类型,从而产生传出类型状态。
打开App,查看更多内容
随时随地看视频慕课网APP

相关分类

Java