我创建了2个简单的类来尝试Java MethodHandle -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方法包含强制转换指令a,但方法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


谁能为我解释一下?

最佳答案

您会注意到invokeExact确实从其描述符int返回了()I

  18: invokevirtual #52    // Method java/lang/invoke/MethodHandle.invokeExact:()I


结果,不需要铸造。

它返回int而不是Object的原因是invokeExact中的invoke方法(以及MethodHandle)被特殊对待,请参见invokeExact API


  返回值:
  
  使用对象静态表示的签名多态结果


并来自signature polymorphism


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

关于java - 为什么(int)MethodHandle.invokeExact不存在checkcast指令?,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/36743385/

10-11 04:33