Java Lambda及InvokedDynamic调用探秘(一

2018-01-03  本文已影响30人  gancheng_wxh

我们先来看看一个java Lambda的简单实例。

package com.github.wanggancheng;
import java.util.function.Consumer;
public class LambdaDemo {


    public static void main(String[] args) {

        Consumer<String> consumer=(s)->System.out.println(s);
        consumer.accept("Lambda demo");
    }
}

这个程序编译后运行时会输出“Lambda demo"。我们先来通过命令查看class字节码文件:

javap  -p LambdaDemo

可以看到此类有下列方法:

public class com.github.wanggancheng.LambdaDemo {
  public com.github.wanggancheng.LambdaDemo();
  public static void main(java.lang.String[]);
  private static void lambda$main$0(java.lang.String);
}

除了类的构造函数及main方法外,多了一个名为lambda$main$0的静态方法。此静态方法的参数为java.lang.String类型。

下面我们通过对字节码的分析来理解背后的密码。

让我们来看看main方法的字节码吧。

  0: invokedynamic #2,  0              // InvokeDynamic #0:accept:()Ljava/util/function/Consumer;
         5: astore_1
         6: aload_1
         7: ldc           #3                  // String Lambda demo
         9: invokeinterface #4,  2            // InterfaceMethod java/util/function/Consumer.accept:(Ljava/lang/Object;)V
        14: return

从offset为0开始,第一条字节码信息为:CONSTANT__InvokeDynamic_info。它在常量池中的位置为2(#2)

  #2 = InvokeDynamic      #0:#36         // #0:accept:()Ljava/util/function/Consumer;

它主要由两部分组成。
第一部分为bootstrap_method_attr_index,也就是指定了bootstrap_methods[]数组中的合法索引。在这里的#0表示指向了bootstrap_methods数组中第1个bootstrap_method。
第二部分为name_and_type_index。#36表示常量池中的位置。

  #36 = NameAndType        #49:#50        // accept:()Ljava/util/function/Consumer;

"accept"是invokedName,"()Ljava/util/function/Consumer"是invokedType。二者合为NameAndType。

接下来,我们看看bootstrap_methods。它的结构定义如下:_

BootstrapMethods_attribute {
u2 attribute_name_index;
u4 attribute_length;
u2 num_bootstrap_methods;
{ u2 bootstrap_method_ref;
u2 num_bootstrap_arguments;
u2 bootstrap_arguments[num_bootstrap_arguments];
} bootstrap_methods[num_bootstrap_methods];
}

每个bootstrap_method由两部分组成:bootstrap_method_ref及bootstrap_arguments。_

在这个类中,bootstrapmethods如下:

BootstrapMethods:
  0: #32 invokestatic java/lang/invoke/LambdaMetafactory.metafactory:(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite;
    Method arguments:
      #33 (Ljava/lang/Object;)V
      #34 invokestatic com/github/wanggancheng/LambdaDemo.lambda$main$0:(Ljava/lang/String;)V
      #35 (Ljava/lang/String;)V

从上面可知,bootstrap_method_ref为#32,它在常量池的内容如下:

#30 = NameAndType #9:#10 // "<init>":()V
#31 = Utf8 BootstrapMethods
#32 = MethodHandle #6:#46 // invokestatic java/lang/invoke/LambdaMetafactory.metafactory:(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite;
#33 = MethodType #47 // (Ljava/lang/Object;)V
#34 = MethodHandle #6:#48 // invokestatic com/github/wanggancheng/LambdaDemo.lambda$main$0:(Ljava/lang/String;)V
#35 = MethodType #25 // (Ljava/lang/String;)V
#36 = NameAndType #49:#50 // accept:()Ljava/util/function/Consumer;
#37 = Utf8 Lambda demo
#38 = Class #51 // java/util/function/Consumer
#39 = NameAndType #49:#47 // accept:(Ljava/lang/Object;)V
#40 = Class #52 // java/lang/System
#41 = NameAndType #53:#54 // out:Ljava/io/PrintStream;
#42 = Class #55 // java/io/PrintStream
#43 = NameAndType #56:#25 // println:(Ljava/lang/String;)V
#44 = Utf8               com/github/wanggancheng/LambdaDemo
#45 = Utf8               java/lang/Object
#46 = Methodref          #57.#58        // java/lang/invoke/LambdaMetafactory.metafactory:(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite;
#47 = Utf8               (Ljava/lang/Object;)V
#48 = Methodref          #7.#59         // com/github/wanggancheng/LambdaDemo.lambda$main$0:(Ljava/lang/String;)V
#49 = Utf8               accept
#50 = Utf8               ()Ljava/util/function/Consumer;
#51 = Utf8               java/util/function/Consumer
#52 = Utf8               java/lang/System
#53 = Utf8               out
#54 = Utf8               Ljava/io/PrintStream;
#55 = Utf8               java/io/PrintStream
#56 = Utf8               println
#57 = Class              #60            // java/lang/invoke/LambdaMetafactory
#58 = NameAndType        #61:#65        // metafactory:(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite;
#59 = NameAndType        #24:#25        // lambda$main$0:(Ljava/lang/String;)V
#60 = Utf8               java/lang/invoke/LambdaMetafactory
#61 = Utf8               metafactory
#62 = Class              #67            // java/lang/invoke/MethodHandles$Lookup
#63 = Utf8               Lookup
#64 = Utf8               InnerClasses
#65 = Utf8               (Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite;

其实是一个MethodHandle。一个MethodHandle主要由方法句柄类型reference_kind及对应的引用类型reference_index。在这里引用类型为invoke_static,引用索引则响应的指向一个静态方法。这个静态方法为_/LambdaMetafactory.metafactory。

第1个bootstrap_method需要三个参数,分别为#33,#34,#35指向的类型。这是与LambdaMetaFactory.metafactory的最后三个参数一致的。

从第1个bootstrap_method来看,最终会调用到编译器自动新增的方法:com/github/wanggancheng/LambdaDemo.lambda$main$0。

那么,最终是如何调用到这个静态方法内。在运行是添加一个参数:-Djdk.internal.lambda.dumpProxyClasses

在执行过程中会生成名为com.github.wanggancheng.LambdaDemo$$Lambda$1的class文件。反编译此类的内容如下:

package com.github.wanggancheng;

import java.lang.invoke.LambdaForm.Hidden;
import java.util.function.Consumer;

// $FF: synthetic class
final class LambdaDemo$$Lambda$1 implements Consumer {
    private LambdaDemo$$Lambda$1() {
    }

    @Hidden
    public void accept(Object var1) {
        LambdaDemo.lambda$main$0((String)var1);
    }
}

也就是说,consumer.accept("Lambda demo")执行过程中最终会调用到LambdaDemo$$Lambda$1的accept方法,方法中会调用到前面说到的自动新增的方法的。

上一篇下一篇

猜你喜欢

热点阅读