日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程语言 > java >内容正文

java

这是一个有趣的问题,Java 8 Lambda 表达式被编译成了什么?

發(fā)布時(shí)間:2025/3/21 java 37 豆豆
生活随笔 收集整理的這篇文章主要介紹了 这是一个有趣的问题,Java 8 Lambda 表达式被编译成了什么? 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

在了解了Java 8 Lambda的一些基本概念和應(yīng)用后, 我們會(huì)有這樣的一個(gè)問題:?Lambda表達(dá)式被編譯成了什么?

這是一個(gè)有趣的問題,涉及到JDK的具體的實(shí)現(xiàn)。本文將介紹OpenJDK對(duì)Lambda表達(dá)式的轉(zhuǎn)換細(xì)節(jié), 讀者可以了解Java 8 Lambda表達(dá)式背景知識(shí)。

Lambda表達(dá)式的轉(zhuǎn)換策略

Brian Goetz是Oracle的Java語言架構(gòu)師, JSR 335(Lambda Expression)規(guī)范的lead, 寫了幾篇Lambda設(shè)計(jì)方面的文章, 其中之一就是Translation of Lambda Expressions。這篇文章介紹了Java 8 Lambda設(shè)計(jì)時(shí)的考慮以及實(shí)現(xiàn)方法。

他提到, Lambda表達(dá)式可以通過內(nèi)部類, method handle, dynamic proxy等方式實(shí)現(xiàn), 但是這些方法各有優(yōu)劣。真正要實(shí)現(xiàn)Lambda表達(dá)式, 必須兼顧兩個(gè)目標(biāo):一是不引入特定策略,以期為將來的優(yōu)化提供最大的靈活性, 二是保持類文件格式的穩(wěn)定。通過Java 7中引入的invokedynamic?(JSR 292), 可以很好的兼顧這兩個(gè)目標(biāo)。

invokedynamic?在缺乏靜態(tài)類型信息的情況下可以支持有效的靈活的方法調(diào)用。主要是為了日益增長(zhǎng)的運(yùn)行在JVM上的動(dòng)態(tài)類型語言, 如Groovy, JRuby。

invokedynamic?將Lambda表達(dá)式的轉(zhuǎn)換策略推遲到運(yùn)行時(shí), 這也意味著我們現(xiàn)在編譯的代碼在將來的轉(zhuǎn)換策略改變的情況下也能正常運(yùn)行。

編譯器在編譯的時(shí)候, 會(huì)將Lambda表達(dá)式的表達(dá)式體 (lambda body)脫糖(desugar) 成一個(gè)方法,此方法的參數(shù)列表和返回類型和lambda表達(dá)式一致, 如果有捕獲參數(shù), 脫糖的方法的參數(shù)可能會(huì)更多一些, 并會(huì)產(chǎn)生一個(gè)invokedynamic調(diào)用, 調(diào)用一個(gè)call site。

這個(gè)call site被調(diào)用時(shí)會(huì)返回lambda表達(dá)式的目標(biāo)類型(functional interface)的一個(gè)實(shí)現(xiàn)類。這個(gè)call site稱為這個(gè)lambda表達(dá)式的lambda factory。?lambda factory的bootstrap方法是一個(gè)標(biāo)準(zhǔn)方法, 叫做lambda metafactory。

編譯器在轉(zhuǎn)換lambda表達(dá)式時(shí), 可以推斷出表達(dá)式的參數(shù)類型,返回類型以及異常, 稱之為natural signature, 我們將目標(biāo)類型的方法簽名稱之為lambda descriptor, lambda factory的返回對(duì)象實(shí)現(xiàn)了函數(shù)式接口, 并且關(guān)聯(lián)的表達(dá)式的代碼邏輯, 稱之為lambda object。

轉(zhuǎn)換舉例

以上的解釋有點(diǎn)晦澀, 簡(jiǎn)單來說

  • 編譯時(shí)

    • Lambda 表達(dá)式會(huì)生成一個(gè)方法, 方法實(shí)現(xiàn)了表達(dá)式的代碼邏輯

    • 生成invokedynamic指令, 調(diào)用bootstrap方法, 由java.lang.invoke.LambdaMetafactory.metafactory方法實(shí)現(xiàn)

  • 運(yùn)行時(shí)

    • invokedynamic指令調(diào)用metafactory方法。它會(huì)返回一個(gè)CallSite, 此CallSite返回目標(biāo)類型的一個(gè)匿名實(shí)現(xiàn)類, 此類關(guān)聯(lián)編譯時(shí)產(chǎn)生的方法

    • lambda表達(dá)式調(diào)用時(shí)會(huì)調(diào)用匿名實(shí)現(xiàn)類關(guān)聯(lián)的方法。

最簡(jiǎn)單的一個(gè)lambda表達(dá)式的例子:

public?class?Lambda1?{public?static?void?main(String[]?args)?{Consumer<String>?c?=?s?->?System.out.println(s);c.accept("hello?lambda");} }

使用javap查看生成的字節(jié)碼?javap -c -p -v com/colobu/lambda/chapter5/Lambda1.class:

[root@colobu?bin]#?javap?-c?-p?-v?com/colobu/lambda/chapter5/Lambda1.class? Classfile?/mnt/eclipse/Lambda/bin/com/colobu/lambda/chapter5/Lambda1.classLast?modified?Nov?6,?2014;?size?1401?bytesMD5?checksum?fe2b2d3f039a9ba4209c488a8c4b4ea8Compiled?from?"Lambda1.java" public?class?com.colobu.lambda.chapter5.Lambda1SourceFile:?"Lambda1.java"BootstrapMethods:0:?#57?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:#58?(Ljava/lang/Object;)V#61?invokestatic?com/colobu/lambda/chapter5/Lambda1.lambda$0:(Ljava/lang/String;)V#62?(Ljava/lang/String;)VInnerClasses:public?static?final?#68=?#64?of?#66;?//Lookup=class?java/lang/invoke/MethodHandles$Lookup?of?class?java/lang/invoke/MethodHandlesminor?version:?0major?version:?52flags:?ACC_PUBLIC,?ACC_SUPER Constant?pool:#1?=?Class??????????????#2?????????????//??com/colobu/lambda/chapter5/Lambda1#2?=?Utf8???????????????com/colobu/lambda/chapter5/Lambda1#3?=?Class??????????????#4?????????????//??java/lang/Object#4?=?Utf8???????????????java/lang/Object#5?=?Utf8???????????????<init>#6?=?Utf8???????????????()V#7?=?Utf8???????????????Code#8?=?Methodref??????????#3.#9??????????//??java/lang/Object."<init>":()V#9?=?NameAndType????????#5:#6??????????//??"<init>":()V#10?=?Utf8???????????????LineNumberTable#11?=?Utf8???????????????LocalVariableTable#12?=?Utf8???????????????this#13?=?Utf8???????????????Lcom/colobu/lambda/chapter5/Lambda1;#14?=?Utf8???????????????main#15?=?Utf8???????????????([Ljava/lang/String;)V#16?=?NameAndType????????#17:#18????????//??accept:()Ljava/util/function/Consumer;#17?=?Utf8???????????????accept#18?=?Utf8???????????????()Ljava/util/function/Consumer;#19?=?InvokeDynamic??????#0:#16?????????//??#0:accept:()Ljava/util/function/Consumer;#20?=?String?????????????#21????????????//??hello?lambda#21?=?Utf8???????????????hello?lambda#22?=?InterfaceMethodref?#23.#25????????//??java/util/function/Consumer.accept:(Ljava/lang/Object;)V#23?=?Class??????????????#24????????????//??java/util/function/Consumer#24?=?Utf8???????????????java/util/function/Consumer#25?=?NameAndType????????#17:#26????????//??accept:(Ljava/lang/Object;)V#26?=?Utf8???????????????(Ljava/lang/Object;)V#27?=?Utf8???????????????args#28?=?Utf8???????????????[Ljava/lang/String;#29?=?Utf8???????????????c#30?=?Utf8???????????????Ljava/util/function/Consumer;#31?=?Utf8???????????????LocalVariableTypeTable#32?=?Utf8???????????????Ljava/util/function/Consumer<Ljava/lang/String;>;#33?=?Utf8???????????????lambda$0#34?=?Utf8???????????????(Ljava/lang/String;)V#35?=?Fieldref???????????#36.#38????????//??java/lang/System.out:Ljava/io/PrintStream;#36?=?Class??????????????#37????????????//??java/lang/System#37?=?Utf8???????????????java/lang/System#38?=?NameAndType????????#39:#40????????//??out:Ljava/io/PrintStream;#39?=?Utf8???????????????out#40?=?Utf8???????????????Ljava/io/PrintStream;#41?=?Methodref??????????#42.#44????????//??java/io/PrintStream.println:(Ljava/lang/String;)V#42?=?Class??????????????#43????????????//??java/io/PrintStream#43?=?Utf8???????????????java/io/PrintStream#44?=?NameAndType????????#45:#34????????//??println:(Ljava/lang/String;)V#45?=?Utf8???????????????println#46?=?Utf8???????????????s#47?=?Utf8???????????????Ljava/lang/String;#48?=?Utf8???????????????SourceFile#49?=?Utf8???????????????Lambda1.java#50?=?Utf8???????????????BootstrapMethods#51?=?Methodref??????????#52.#54????????//??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;#52?=?Class??????????????#53????????????//??java/lang/invoke/LambdaMetafactory#53?=?Utf8???????????????java/lang/invoke/LambdaMetafactory#54?=?NameAndType????????#55:#56????????//??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;#55?=?Utf8???????????????metafactory#56?=?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;#57?=?MethodHandle???????#6:#51?????????//??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;#58?=?MethodType?????????#26????????????//??(Ljava/lang/Object;)V#59?=?Methodref??????????#1.#60?????????//??com/colobu/lambda/chapter5/Lambda1.lambda$0:(Ljava/lang/String;)V#60?=?NameAndType????????#33:#34????????//??lambda$0:(Ljava/lang/String;)V#61?=?MethodHandle???????#6:#59?????????//??invokestatic?com/colobu/lambda/chapter5/Lambda1.lambda$0:(Ljava/lang/String;)V#62?=?MethodType?????????#34????????????//??(Ljava/lang/String;)V#63?=?Utf8???????????????InnerClasses#64?=?Class??????????????#65????????????//??java/lang/invoke/MethodHandles$Lookup#65?=?Utf8???????????????java/lang/invoke/MethodHandles$Lookup#66?=?Class??????????????#67????????????//??java/lang/invoke/MethodHandles#67?=?Utf8???????????????java/lang/invoke/MethodHandles#68?=?Utf8???????????????Lookup {public?com.colobu.lambda.chapter5.Lambda1();flags:?ACC_PUBLICCode:stack=1,?locals=1,?args_size=10:?aload_0???????1:?invokespecial?#8??????????????????//?Method?java/lang/Object."<init>":()V4:?return????????LineNumberTable:line?7:?0LocalVariableTable:Start??Length??Slot??Name???Signature0???????5?????0??this???Lcom/colobu/lambda/chapter5/Lambda1;public?static?void?main(java.lang.String[]);flags:?ACC_PUBLIC,?ACC_STATICCode:stack=2,?locals=2,?args_size=10:?invokedynamic?#19,??0?????????????//?InvokeDynamic?#0:accept:()Ljava/util/function/Consumer;5:?astore_1??????6:?aload_1???????7:?ldc???????????#20?????????????????//?String?hello?lambda9:?invokeinterface?#22,??2???????????//?InterfaceMethod?java/util/function/Consumer.accept:(Ljava/lang/Object;)V14:?return????????LineNumberTable:line?10:?0line?11:?6line?12:?14LocalVariableTable:Start??Length??Slot??Name???Signature0??????15?????0??args???[Ljava/lang/String;6???????9?????1?????c???Ljava/util/function/Consumer;LocalVariableTypeTable:Start??Length??Slot??Name???Signature6???????9?????1?????c???Ljava/util/function/Consumer<Ljava/lang/String;>;private?static?void?lambda$0(java.lang.String);flags:?ACC_PRIVATE,?ACC_STATIC,?ACC_SYNTHETICCode:stack=2,?locals=1,?args_size=10:?getstatic?????#35?????????????????//?Field?java/lang/System.out:Ljava/io/PrintStream;3:?aload_0???????4:?invokevirtual?#41?????????????????//?Method?java/io/PrintStream.println:(Ljava/lang/String;)V7:?return????????LineNumberTable:line?10:?0LocalVariableTable:Start??Length??Slot??Name???Signature0???????8?????0?????s???Ljava/lang/String; }

可以看到, Lambda表達(dá)式體被生成一個(gè)稱之為lambda$0的方法??醋止?jié)碼知道它調(diào)用System.out.println輸出傳入的參數(shù)。

原lambda表達(dá)式處產(chǎn)生了一條invokedynamic #19, 0。它會(huì)調(diào)用bootstrap方法。

0:?#57?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:#58?(Ljava/lang/Object;)V#61?invokestatic?com/colobu/lambda/chapter5/Lambda1.lambda$0:(Ljava/lang/String;)V#62?(Ljava/lang/String;)V

如果Lambda表達(dá)式寫成Consumer<String> c = (Consumer<String> & Serializable)s -> System.out.println(s);, 則BootstrapMethods的字節(jié)碼為

BootstrapMethods:0:?#108?invokestatic?java/lang/invoke/LambdaMetafactory.altMetafactory:(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;[Ljava/lang/Object;)Ljava/lang/invoke/CallSite;Method?arguments:#109?(Ljava/lang/Object;)V#112?invokestatic?com/colobu/lambda/chapter5/Lambda1.lambda$0:(Ljava/lang/String;)V#113?(Ljava/lang/String;)V#114?1

它調(diào)用的是LambdaMetafactory.altMetafactory,和上面的調(diào)用的方法不同。#114 1意味著要實(shí)現(xiàn)Serializable接口。

如果Lambda表達(dá)式寫成``,則BootstrapMethods的字節(jié)碼為

BootstrapMethods:0:?#57?invokestatic?java/lang/invoke/LambdaMetafactory.altMetafactory:(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;[Ljava/lang/Object;)Ljava/lang/invoke/CallSite;Method?arguments:#58?(Ljava/lang/Object;)V#61?invokestatic?com/colobu/lambda/chapter5/Lambda1.lambda$0:(Ljava/lang/String;)V#62?(Ljava/lang/String;)V#63?2#64?1#65?com/colobu/lambda/chapter5/ABC

#63 2意味著要實(shí)現(xiàn)額外的接口。#64 1意味著要實(shí)現(xiàn)額外的接口的數(shù)量為1。

字節(jié)碼的指令含義可以參考這篇文章:Java bytecode instruction listings。

可以看到, Lambda表達(dá)式具體的轉(zhuǎn)換是通過java.lang.invoke.LambdaMetafactory.metafactory實(shí)現(xiàn)的, 靜態(tài)參數(shù)依照lambda表達(dá)式和目標(biāo)類型不同而不同。

LambdaMetafactory.metafactory

現(xiàn)在我們可以重點(diǎn)關(guān)注以下?LambdaMetafactory.metafactory的實(shí)現(xiàn)。

public?static?CallSite?metafactory(MethodHandles.Lookup?caller,String?invokedName,MethodType?invokedType,MethodType?samMethodType,MethodHandle?implMethod,MethodType?instantiatedMethodType)throws?LambdaConversionException?{返回值類型AbstractValidatingLambdaMetafactory?mf;mf?=?new?InnerClassLambdaMetafactory(caller,?invokedType,invokedName,?samMethodType,implMethod,?instantiatedMethodType,false,?EMPTY_CLASS_ARRAY,?EMPTY_MT_ARRAY);mf.validateMetafactoryArgs();return?mf.buildCallSite();}

實(shí)際是由InnerClassLambdaMetafactory的buildCallSite來生成。生成之前會(huì)調(diào)用validateMetafactoryArgs方法校驗(yàn)?zāi)繕?biāo)類型(SAM)方法的參數(shù)/和產(chǎn)生的方法的參數(shù)/返回值類型是否一致。

metaFactory方法的參數(shù):

  • caller: 由JVM提供的lookup context

  • invokedName: JVM提供的NameAndType

  • invokedType: JVM提供的期望的CallSite類型

  • samMethodType: 函數(shù)式接口定義的方法的簽名

  • implMethod: 編譯時(shí)產(chǎn)生的那個(gè)實(shí)現(xiàn)方法

  • instantiatedMethodType: 強(qiáng)制的方法簽名和返回類型, 一般和samMethodType相同或者是它的一個(gè)特例

上面的代碼基本上是InnerClassLambdaMetafactory.buildCallSite的包裝,下面看看這個(gè)方法的實(shí)現(xiàn):

CallSite?buildCallSite()?throws?LambdaConversionException?{final?Class<?>?innerClass?=?spinInnerClass();if?(invokedType.parameterCount()?==?0)?{.....?//調(diào)用構(gòu)造函數(shù)初始化一個(gè)SAM的實(shí)例return?new?ConstantCallSite(MethodHandles.constant(samBase,?inst));}?else?{UNSAFE.ensureClassInitialized(innerClass);return?new?ConstantCallSite(MethodHandles.Lookup.IMPL_LOOKUP.findStatic(innerClass,?NAME_FACTORY,?invokedType));}}

其中spinInnerClass調(diào)用asm框架動(dòng)態(tài)的產(chǎn)生SAM的實(shí)現(xiàn)類, 這個(gè)實(shí)現(xiàn)類的的方法將會(huì)調(diào)用編譯時(shí)產(chǎn)生的那個(gè)實(shí)現(xiàn)方法。你可以在編譯的時(shí)候加上參數(shù)-Djdk.internal.lambda.dumpProxyClasses, 這樣編譯的時(shí)候會(huì)自動(dòng)產(chǎn)生運(yùn)行時(shí)spinInnerClass產(chǎn)生的類。你可以訪問OpenJDK的bug系統(tǒng)了解這個(gè)功能。?JDK-8023524

重復(fù)的lambda表達(dá)式

下面的代碼中,在一個(gè)循環(huán)中重復(fù)生成調(diào)用lambda表達(dá)式,只會(huì)生成同一個(gè)lambda對(duì)象, 因?yàn)橹挥型粋€(gè)invokedynamic指令。

for?(int?i?=?0;?i<100;?i++){Consumer<String>?c?=?s?->?System.out.println(s);System.out.println(c.hashCode()); }

但是下面的代碼會(huì)生成兩個(gè)lambda對(duì)象, 因?yàn)樗鼤?huì)生成兩個(gè)invokedynamic指令。

Consumer<String>?c?=?s?->?System.out.println(s); System.out.println(c.hashCode()); Consumer<String>?c2?=?s?->?System.out.println(s); System.out.println(c2.hashCode());

生成的類名

既然LambdaMetafactory會(huì)使用asm框架生成一個(gè)匿名類, 那么這個(gè)類的類名有什么規(guī)律的。

Consumer<String>?c?=?s?->?System.out.println(s); System.out.println(c.getClass().getName()); System.out.println(c.getClass().getSimpleName()); System.out.println(c.getClass().getCanonicalName());

輸出結(jié)果如下:

com.colobu.lambda.chapter5.Lambda3$$Lambda$1/640070680 Lambda3$$Lambda$1/640070680 com.colobu.lambda.chapter5.Lambda3$$Lambda$1/640070680

類名格式如 <包名>.<類名>$$Lambda$/. number是由一個(gè)計(jì)數(shù)器生成counter.incrementAndGet()。后綴/<NN>中的數(shù)字是一個(gè)hash值, 那就是類對(duì)象的hash值c.getClass().hashCode()。在Klass::external_name()中生成。

sprintf(hash_buf,?"/"?UINTX_FORMAT,?(uintx)hash);

直接調(diào)用生成的方法

上面提到, Lambda表達(dá)式體會(huì)由編譯器生成一個(gè)方法,名字格式如Lambda$XXX。既然是類中的實(shí)實(shí)在在的方法,我們就可以直接調(diào)用。當(dāng)然, 你在代碼中直接寫lambda$0()編譯通不過, 因?yàn)長(zhǎng)ambda表達(dá)式體還沒有被抽取成方法。但是在運(yùn)行中我們可以通過反射的方式調(diào)用。下面的例子使用發(fā)射和MethodHandle兩種方式調(diào)用這個(gè)方法。

public?static?void?main(String[]?args)?throws?Throwable?{Consumer<String>?c?=?s?->?System.out.println(s);Method?m?=?Lambda4.class.getDeclaredMethod("lambda$0",?String.class);m.invoke(null,?"hello?reflect");MethodHandle?mh?=?MethodHandles.lookup().findStatic(Lambda4.class,?"lambda$0",?MethodType.methodType(void.class,?String.class));mh.invoke("hello?MethodHandle"); }

捕獲的變量等價(jià)于'final'

我們知道,在匿名類中調(diào)用外部的參數(shù)時(shí),參數(shù)必須聲明為final。Lambda體內(nèi)也可以引用上下文中的變量,變量可以不聲明成final的,但是必須等價(jià)于final。下面的例子中變量capturedV等價(jià)與final, 并沒有在上下文中重新賦值。

public?class?Lambda5?{String?greeting?=?"hello";public?static?void?main(String[]?args)?throws?Throwable?{Lambda5?capturedV?=?new?Lambda5();Consumer<String>?c?=?s?->?System.out.println(capturedV.greeting?+?"?"?+?s);c.accept("captured?variable");//capturedV?=?null;?//Local?variable?capturedV?defined?in?an?enclosing?scope?must?be?final?or?effectively?final//capturedV.greeting?=?"hi";} }

如果反注釋capturedV = null;編譯出錯(cuò),因?yàn)閏apturedV在上下文中被改變。但是如果反注釋capturedV.greeting = "hi";?則沒問題, 因?yàn)閏apturedV沒有被重新賦值, 只是它指向的對(duì)象的屬性有所變化。

方法引用

public?static?void?main(String[]?args)?throws?Throwable?{Consumer<String>?c??=?System.out::println;c.accept("hello"); }

這段代碼不會(huì)產(chǎn)生一個(gè)類似"Lambda$0"新方法。因?yàn)長(zhǎng)ambdaMetafactory會(huì)直接使用這個(gè)引用的方法。

BootstrapMethods:0:?#51?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:#52?(Ljava/lang/Object;)V#59?invokevirtual?java/io/PrintStream.println:(Ljava/lang/String;)V#60?(Ljava/lang/String;)V

#59指示實(shí)現(xiàn)方法為System.out::println

總結(jié)

以上是生活随笔為你收集整理的这是一个有趣的问题,Java 8 Lambda 表达式被编译成了什么?的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。