成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

和Lambdas的第一次親密接觸

開發 后端
在這篇文章里,我的目的是要看看面紗后的東西 ——看看在運行時環境里lambdas是表現的,在方法的調度過程中涉及到哪些字節碼指令。

java 8

Lambda工程是即將到來的Java8的一大主題,可能也是程序員們最期待已久的東西。隨著Java lambdas的到來,還有一個有趣的東西被附帶的加進了Java語言——defender(守衛者)方法。在這篇文章里,我的目的是要看看面紗后的東西 ——看看在運行時環境里lambdas是表現的,在方法的調度過程中涉及到哪些字節碼指令。

盡管Java 8還沒有正式發布,我們仍然可以下載各種平臺上的早期預覽版,在其上做簡單的嘗試。

你也想試試lambdas,是嗎?

如果你熟悉其它的還有lambda表達式的編程語言,比如Groovy 或 Ruby,當第一眼看到Java里的lambda時,你也許會吃驚于它的不簡單。在Java里,lambda表達式是“SAM”(Single Abstract Method)——一個含有一個抽象方法的接口(是的,現在接口里可以含有一個非抽象的方法,defender守衛方法)。

舉個例子,大家熟知的Runnable接口就可以完美的被當作一個SAM類型:

  1. Runnable r = () -> System.out.println("hello lambda!"); 

,這同樣也適用于Comparable接口:

  1. Comparator<Integer> cmp = (x, y) -> (x < y) ? -1 : ((x > y) ? 1 : 0); 

寫成下面的樣子也是一樣的:

  1. Comparator<Integer> cmp = (x, y) -> {  
  2.   return (x < y) ? -1 : ((x > y) ? 1 : 0);  
  3. }; 

從中可以看出,單行的lambda表達式似乎是隱含了一個return語句。

那么,如何寫一個能接受lambda表達式作為參數的方法呢?這樣,你需要先把這個參數聲明成函數式的接口,然后把lambda傳入:

  1. interface Action {  
  2.    void run(String param);  
  3. }  
  4.  
  5. public void execute(Action action){  
  6.    action.run("Hello!");  

一旦有了一個能將函數式接口作為參數的方法,我們就可以像下面這樣調用它:

  1. execute((String s) -> System.out.println(s)); 

還可以更簡潔,這個表達式可以被替換成對一個方法的引用,因為它只是單個方法,而且它們的參數是相同的:

  1. execute(System.out::println); 

然而,如果參數上有任何其它形式的變化,我們就不能直接引用方法,必須寫全lambda表達式:

  1. execute((String s) -> System.out.println("*" + s + "*")); 

我覺得這種語法還是相當漂亮的,現在,Java語言里有了一個非常優雅的lambdas解決方案,盡管Java里并不存在函數式類型。

JDK 8里的函數式接口

我們已經知道,lambda在運行時的表現形式是一個函數式的接口(或“SAM類型”)——只有一個抽象方法的接口。盡管JDK里已經有了不少這樣的接口,例如RunnableComparable ,它們符合這種標準,但很顯然,對于一個新API的進化來說,這是不夠的。我們不可能所有地方都用Runnables接口。

在JDK 8 里有個新包,java.util.function,里面包含了很多函數式接口,都是提供在新API里使用的。我不想把它們全列出來——你們自己可以去看一下,學習一下這個新包 [[64637]]

但看起來這個新包在不斷的變化,經常性的一些新接口會出現而另一些會消失。例如,以前曾有過 java.util.function.Block 這個類,最新的版本中卻沒有它,我寫這篇博客時使用的版本是:

  1. anton$ java -version  
  2. openjdk version "1.8.0-ea" 
  3. OpenJDK Runtime Environment (build 1.8.0-ea-b75)  
  4. OpenJDK 64-Bit Server VM (build 25.0-b15, mixed mode) 

我研究發現,它現在被 Consumer 接口替代,collection包里的所有新方法都將使用它。例如,Collection接口里定義了forEach方法,如下:

  1. public default void forEach(Consumer<? super T> consumer) {  
  2.   for (T t : this) {  
  3.     consumer.accept(t);  
  4.   }  

Consumer接口里一個有趣地方是,它實際上定義了一個抽象方法——accept(T t)和一個defender方法——Consumer<T> chain(Consumer<? extend T> consumer)。這就是說你可以鏈式調用這個接口。我不確定如何使用,因為我在JDK包里沒有找到chain(..)的使用方法說明。

我還發現所有的接口都使用了@FunctionalInterface運行時注注解注釋。這個注釋不僅僅是個說明,它還被javac使用來驗證這個接口是否真是一個函數式接口,是否至少有一個抽象方法在里面。

所以,如果我們來編譯下面的這段代碼

  1. @FunctionalInterface 
  2. interface Action {  
  3.   void run(String param);  
  4.   void stop(String param);  

編譯器會告訴我們:

  1. java: Unexpected @FunctionalInterface annotation  
  2.   Action is not a functional interface 
  3.     multiple non-overriding abstract methods found in interface Action 

而下面的就能編譯通過:

  1. @FunctionalInterface 
  2. interface Action {  
  3.   void run(String param);  
  4.   default void stop(String param){}  

反編譯lambdas

我對語法語言特征其實并不是很好奇,我更好奇的是這些特征在運行時的表現形式,這就是為什么我像往常一樣,拿起我喜愛的javap工具,開始查看lambdas里的這些類的字節碼

目前(在Java 7之前),如果你想在Java里模擬lambdas,你需要定義一個匿名的內部類。它在編譯后會產生一個具體的class。如果你在一段代碼里定義了多個這樣的類,你會發現這些類后面會跟著一些數字。那lambdas也會這樣嗎?

看看下面的這段代碼:

  1. public class Main {  
  2.  
  3.   @FunctionalInterface 
  4.   interface Action {  
  5.     Object run(String s);  
  6.   }  
  7.  
  8.   public void action(Action action){  
  9.     action.run("Hello!");  
  10.   }  
  11.  
  12.   public static void main(String[] args) {  
  13.     new Main().action((String s) -> System.out.print("*" + s + "*"));  
  14.   }  
  15.  

編譯產生了兩個類文件:Main.classMain$Action.class,沒有匿名類實現里那樣的序號化的類。那么,在Main.class里應該會有一些東西來代表我在main方法里定義的lambdas表達式的實現。

  1. $ javap -p Main   
  2.  
  3. Warning: Binary file Main contains com.zt.Main  
  4. Compiled from "Main.java" 
  5. public class com.zt.Main {  
  6.   public com.zt.Main();  
  7.   public void action(com.zt.Main$Action);  
  8.   public static void main(java.lang.String[]);  
  9.   private static java.lang.Object lambda$0(java.lang.String);  

啊哈!編譯類了產生了lambda$0方法!使用-c -v指示符會讓我們看到真正的字節碼,以及常量池的定義。

main方法里顯示,invokedynamic被用來調度這個調用:

  1. public static void main(java.lang.String[]);  
  2.   Code:  
  3.    0new               #4    // class com/zt/Main  
  4.    3: dup            
  5.    4: invokespecial #5        // Method "":()V  
  6.    7: invokedynamic #6,  0    // InvokeDynamic #0:lambda:()Lcom/zt/Main$Action;  
  7.    12: invokevirtual #7       // Method action:(Lcom/zt/Main$Action;)V  
  8.    15return 

而在常量池里,你也可以找到運行時的啟動方法:

  1. BootstrapMethods:  
  2.   0: #40 invokestatic java/lang/invoke/LambdaMetafactory.metaFactory:(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite;  
  3.   Method arguments:  
  4.     #41 invokeinterface com/zt/Main$Action.run:(Ljava/lang/String;)Ljava/lang/Object;  
  5.     #42 invokestatic com/zt/Main.lambda$0:(Ljava/lang/String;)Ljava/lang/Object;  
  6.     #43 (Ljava/lang/String;)Ljava/lang/Object; 

你會發現到處都是在使用MethodHandle API,但我們現在不打算深入到里面。現在我們可以確認一點,我們的定義是引用了編譯出來lambda$0方法。

我很好奇,如果我定義一個相同名字的靜態方法會怎樣——畢竟“lambda$0”是一個有效的標識符!于是,我定義了自己的lambda$0方法:

  1. public static Object lambda$0(String s){ return null; } 

而編譯失敗,編譯器不允許我在代碼了擁有這個方法:

  1. java: the symbol lambda$0(java.lang.String) conflicts with a   
  2.            compiler-synthesized symbol in com.zt.Main 

同時,如果我刪掉這段定義lambdas表達式的代碼,程序能順利編譯通過。這就是說,lambdas表達式在編譯期間會比類里的其它數據早先分析,不過這只是我的猜測。

請注意:在這個例子中,lambda并沒有去引用任何變量,也沒有引用類內部的任何方法。這就是為什么產生的lambda$0方法是靜態的。如果lambdas引用了上下文中的變量或方法,那生成的將是一個非靜態方法。所以,不要被這個例子誤導——lambdas是可以捕獲上下文環境內容的!

總結lambdas

我可以毫無疑問的說,lambdas和伴隨它一起的各種特征(守衛方法(defender)
,升級的集合類庫)將很快給Java帶來巨大的沖擊。它的語法相當的簡單,一旦程序員們意識到這些功能給開發效率帶來的好處,我們將會看到大量的程序員都會運用這個功能。

看看lambdas會編譯成什么樣子,這對于我來說是一件非常有趣的事情,我很開心,因為我看到這些所有的invokedynamic指令調用都沒有出現匿名內部類。

英文原文:Java 8: The First Taste of Lambdas

譯文鏈接:http://www.aqee.net/java-8-the-first-taste-of-lambdas/

責任編輯:林師授 來源: 外刊IT評論
相關推薦

2017-08-08 12:50:51

Serverless云端數據庫

2009-11-27 14:18:37

Scala

2010-01-29 09:01:40

.NET 4.0

2018-06-12 11:19:28

2020-05-06 07:18:59

數據中臺架構

2015-11-03 11:39:18

清華大學OpenStackEasyStack

2012-10-19 09:27:23

Ruby

2009-05-15 08:59:32

Windows 7微軟操作系統

2018-08-16 11:30:12

JavaCPU緩存

2010-05-25 13:17:28

MySQL數據庫

2009-06-08 16:00:00

ASP.NET 3.5

2011-07-21 21:01:37

諾基亞塞班蘋果

2017-03-22 15:38:28

代碼架構Java

2013-05-13 11:35:53

獨立開發開發經驗開發感悟

2023-04-12 08:14:10

mysql關聯字段索引

2022-03-16 14:59:28

打包debian模板文件

2012-04-13 10:11:58

Windows 8泄露

2023-09-11 00:14:46

后端團隊項目

2015-08-05 10:54:49

2021-05-24 11:55:37

代碼編碼開發
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 成人在线黄色 | 伊人久久大香线 | 国产成人小视频 | 91日日| 精彩视频一区二区三区 | 国产成人福利在线观看 | 国产一区二区三区精品久久久 | 亚洲精品免费视频 | 久久久久久国产精品三区 | 亚洲 欧美 另类 综合 偷拍 | 精品少妇一区二区三区在线播放 | 亚洲精品一区中文字幕 | 在线观看视频91 | 在线观看中文字幕视频 | 午夜影院在线观看 | 久久亚洲美女 | 精品网 | 91精品国产91久久综合桃花 | 欧美一级特黄aaa大片在线观看 | 亚洲影音先锋 | 99热国产在线播放 | 在线色网| 二区三区视频 | 亚洲激情一级片 | 久久久久久久成人 | 日韩欧美精品 | 欧美一区二区三区久久精品视 | 男人av在线播放 | h视频在线免费观看 | 欧美在线一区二区视频 | 亚洲不卡一| 精品久久久久久久久久久久 | 中文精品视频 | 成人在线精品视频 | 国内精品久久影院 | 国产欧美一区二区精品久导航 | 亚洲欧美精品 | 国产乱码久久久 | 日韩欧美在线观看 | 男人的天堂视频网站 | 视频在线一区二区 |