java - 调用特殊验证错误 : Type is not assignable

标签 java java-8 jvm bytecode verifyerror

我修改了下面字节码的第 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 调用父类(super class)(如 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
            .....
            StackMapTable: number_of_entries = 3
              frame_type = 252 /* append */
                offset_delta = 71
                locals = [ class java/lang/StringBuilder ]
              frame_type = 13 /* same */
              frame_type = 35 /* same */

最佳答案

invokespecial用于实现三件事之一

  1. 构造函数调用
  2. 调用 private方法
  3. 正在做 super. …打电话

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

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

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

相关JVMS规则已在apangin’s answer中引用已经。

关于java - 调用特殊验证错误 : Type is not assignable,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55103524/

相关文章:

java - 在嵌套数据结构上使用 Java8 流创建 map

java - 我如何在 Java 8 lambda 中制作这段代码

JVM 8 更新后 Java Sound 显着变慢

android - Android 中的 DalvikVM 与 JavaVM 的对比?

java - 在 JAVA 中调试多线程 Web 服务器

Java Swing JList,无法设置最小宽度

java - 两个类以不同的方式看待 JDBC 驱动程序

java - 在哪里可以找到 Oracle JVM 的默认 -Xss(堆栈大小)值?

java - 使用 Spring/EHCache 在负载下刷新缓存

java - 如何使用hibernate在spring boot中调用MySQL存储过程?