为什么(int)MethodHandle.invokeExact缺少checkcast-instruction?

时间:2016-04-20 12:11:12

标签: java casting jvm bytecode methodhandle

我为try Java MethodHandle创建了2个简单类--API:

public class Foo {
    private static int staticField;

    public static Object getStaticField() {
        return staticField;
    }
}

以两种方式调用方法Foo.getStaticField()的另一个类 - 直接和使用MethodHandle - API:

....
public static void methodHandleGetStaticField() throws Throwable {
    MethodHandle methodHandle = lookup.findStatic(Foo.class, "getStaticField", MethodType.methodType(int.class));
    int i = (int)methodHandle.invokeExact();
}

public static void directGetStaticField() {
    int i = (int)Foo.getStaticField();
}
....

我已经反编译了类,看到directGetStaticField方法包含强制转换指令,但方法methodHandleGetStaticField没有,但java.lang.invoke.MethodHandle.invokeExact()返回java.lang.Object

public static void directGetStaticField();
descriptor: ()V
Code:
   0: invokestatic  #70    // Method ru/fj/methodhandle/Foo.getStaticField:()Ljava/lang/Object;
   3: checkcast     #33    // class java/lang/Integer
   6: invokevirtual #74    // Method java/lang/Integer.intValue:()I
   9: istore_0
  10: return

public static void methodHandleGetStaticField() throws java.lang.Throwable;
descriptor: ()V
Code:
   0: getstatic     #15    // Field lookup:Ljava/lang/invoke/MethodHandles$Lookup;
   3: ldc           #29    // class ru/fj/methodhandle/Foo
   5: ldc           #90    // String getStaticField
   7: getstatic     #32    // Field java/lang/Integer.TYPE:Ljava/lang/Class;
  10: invokestatic  #38    // Method java/lang/invoke/MethodType.methodType:(Ljava/lang/Class;)Ljava/lang/invoke/MethodType;
  13: invokevirtual #46    // Method java/lang/invoke/MethodHandles$Lookup.findStatic:(Ljava/lang/Class;Ljava/lang/String;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/MethodHandle;
  16: astore_0
  17: aload_0
  18: invokevirtual #52    // Method java/lang/invoke/MethodHandle.invokeExact:()I
  21: istore_1
  22: return

有人可以帮我解释一下吗?

1 个答案:

答案 0 :(得分:3)

您可以注意到description()确实从其描述符invokeExact返回int

()I

因此,不需要铸造。

它返回 18: invokevirtual #52 // Method java/lang/invoke/MethodHandle.invokeExact:()I 而不是int的原因是,Object中的方法invokeExact(以及invoke)被特别处理,请参阅invokeExact API

  

<强>返回

     

签名 - 多态结果,使用Object

静态表示

来自signature polymorphism

  

...不寻常的部分是符号类型描述符是从实际的参数和返回类型派生的,而不是从方法声明派生的。