为了账号安全,请及时绑定邮箱和手机立即绑定

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

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

倚天杖 2022-07-27 21:21:46
我修改了下面字节码的第 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
查看完整描述

2 回答

?
绝地无双

TA贡献1946条经验 获得超4个赞

invokespecial用于实现三件事之一

  1. 构造函数调用

  2. 调用private方法

  3. super. …打电话_

虽然 1. 在这里不适用(因为目标方法的名称不是<init>),但其他任何一种情况都要求接收器类型是当前类或其子类。因此,即使方法的声明类是Type,实际接收者的类型也应该可以分配给当前类,Type$ClassType

与您通过更改创建的最接近的等价物是super调用,尽管在 Java 源代码中,调用方法 viasuper强制接收器引用与 相同this,它本质上可分配给当前类。

在字节码级别,规则限制较少,但不允许在可能指向完全不相关的子类层次结构的实例的类型引用上调用允许绕过当前类或其子类中的方法声明的方法调用,即Type不是一个Type$ClassType

apangin 的回答中已经引用了相关的 JVMS 规则。


查看完整回答
反对 回复 2022-07-27
?
肥皂起泡泡

TA贡献1829条经验 获得超6个赞

您尝试在(@9 返回)invokespecial的实例上执行,而验证器期望当前类的引用,即.TypeinvokevirtualType$ClassType

请参阅 JVMS §4.10.1.9

可以将传入操作数堆栈上与当前类和 Descriptor 中给出的参数类型匹配的类型有效地替换为 Descriptor 中给出的返回类型,从而产生传出类型状态。


查看完整回答
反对 回复 2022-07-27
  • 2 回答
  • 0 关注
  • 140 浏览

添加回答

举报

0/150
提交
取消
微信客服

购课补贴
联系客服咨询优惠详情

帮助反馈 APP下载

慕课网APP
您的移动学习伙伴

公众号

扫描二维码
关注慕课网微信公众号