日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
Java8的Lambda表達式

Java 8預(yù)計將在2013 年發(fā)布,Java 8將支持Lambda功能,盡管該規(guī)范還在不斷的變化,但是Java 8的開發(fā)版已經(jīng)實現(xiàn)了對Lambda的支持。

創(chuàng)新互聯(lián)建站專注于做網(wǎng)站、網(wǎng)站設(shè)計、網(wǎng)頁設(shè)計、網(wǎng)站制作、網(wǎng)站開發(fā)。公司秉持“客戶至上,用心服務(wù)”的宗旨,從客戶的利益和觀點出發(fā),讓客戶在網(wǎng)絡(luò)營銷中找到自己的駐足之地。尊重和關(guān)懷每一位客戶,用嚴謹?shù)膽B(tài)度對待客戶,用專業(yè)的服務(wù)創(chuàng)造價值,成為客戶值得信賴的朋友,為客戶解除后顧之憂。

關(guān)于 Lambda 表達式的定義請看維基百科。

該文章將帶你熟悉 Lambda 語法,以及使用集合 API 中的 Lambda 以及相關(guān)的語言增強,本文所有的代碼都是在 JDK 8 Lambda build b39 編譯。

功能接口

只包含一個方法的接口被稱為功能接口,Lambda 表達式用用于任何功能接口適用的地方。

java.awt.event.ActionListener 就是一個功能接口,因為它只有一個方法:void actionPerformed(ActionEvent). 在 Java 7 中我們會編寫如下代碼:

 
 
 
  1. button.addActionListener(new ActionListener() { 
  2.     public void actionPerformed(ActionEvent e) { 
  3.         ui.dazzle(e.getModifiers());
  4.     }
  5. });

而 Java 8 中可以簡化為:

 
 
 
  1. button.addActionListener(e -> { ui.dazzle(e.getModifiers()); });

編譯器知道Lambda 表達式必須符合 void actionPerformed(ActionEvent) 方法的定義??雌饋?Lambda 實體返回 void,實際上它可以推斷出參數(shù) e 的類型是 java.awt.event.ActionEvent.

函數(shù)集合

Java 8 的類庫包含一個新的包 java.util.functions ,這個包中有很多新的功能接口,這些接口可與集合 API 一起使用。

java.util.functions.Predicate

使用謂詞 (Predicate) 來篩選集合:

 
 
 
  1. List names = Arrays.asList("Alice", "Bob", "Charlie", "Dave");
  2. List filteredNames = names
  3.         .filter(e -> e.length() >= 4)
  4.         .into(new ArrayList());
  5. for (String name : filteredNames) {
  6.     System.out.println(name);
  7. }

這里我們有兩個新方法:

java.util.functions.Block

我們可使用一個新的迭代器方法來替換 for 循環(huán) void forEach(Block):

 
 
 
  1. List names = Arrays.asList("Alice", "Bob", "Charlie", "Dave");
  2. names
  3.    .filter(e -> e.length() >= 4)
  4.    .forEach(e -> { System.out.println(e); });

forEach() 方法是 internal iteration 的一個實例:迭代過程在 Iterable 和 Block 內(nèi)部進行,每次可訪問一個元素。

***的結(jié)果就是用更少的代碼來處理集合:

 
 
 
  1. List names = Arrays.asList("Alice", "Bob", "Charlie", "Dave");
  2. names
  3.    .mapped(e -> { return e.length(); })
  4.    .asIterable() // returns an Iterable of BiValue elements
  5.                  // an element's key is the person's name, its value is the string length
  6.    .filter(e -> e.getValue() >= 4)
  7.    .sorted((a, b) -> a.getValue() - b.getValue())
  8.    .forEach(e -> { System.out.println(e.getKey() + '\t' + e.getValue()); });

這樣做的優(yōu)點是:

元素在需要的時候才進行計算

如果我們?nèi)∫粋€上千個元素的集合的前三條時,其他元素就不會被映射

鼓勵使用方法鏈

我們無需才存儲中間結(jié)果來構(gòu)建新的集合

內(nèi)部迭代過程因此大多數(shù)細節(jié)

例如,我們可以通過下面代碼來并行 map() 操作

writing myCollection.parallel().map(e ?> e.length()).

方法引用

我們可通過 :: 語法來引用某個方法。方法引用被認為是跟 Lambda 表達式一樣的,可用于功能接口所適用的地方。

我們可以引用一個靜態(tài)方法:

 
 
 
  1. executorService.submit(MethodReference::sayHello);
  2. private static void sayHello() {
  3.         System.out.println("hello");
  4. }

或者是一個實例的方法:

 
 
 
  1. Arrays.asList("Alice", "Bob", "Charlie", "Dave").forEach(System.out::println);

我們也可以創(chuàng)建工程方法并將構(gòu)造器引用賦值給 java.util.functions.Factory:

 
 
 
  1. Factory biscuitFactory = Biscuit::new;
  2. Biscuit biscuit = biscuitFactory.make();

***,我們創(chuàng)建一個引用到隨意實例的例子:

 
 
 
  1. interface Accessor {
  2.         PROPERTY access(BEAN bean);
  3. }
  4. public static void main(String[] args) {
  5.         Address address = new Address("29 Acacia Road", "Tunbridge Wells");
  6.         Accessor accessor = Address::getCity;
  7.         System.out.println(accessor.access(address));
  8. }

這里我們無需綁定方法引用到某個實例,我們直接將實例做為功能接口的參數(shù)進行傳遞。

默認方法

直到今天的 Java ,都不可能為一個接口添加方法而不會影響到已有的實現(xiàn)類。而 Java 8 允許你為接口自身指定一個默認的實現(xiàn):

 
 
 
  1. interface Queue {
  2.         Message read();
  3.         void delete(Message message);
  4.         void deleteAll() default {
  5.                 Message message;
  6.                 while ((message = read()) != null) {
  7.                         delete(message);
  8.                 }
  9.         }
  10. }

子接口可以覆蓋默認的方法:

 
 
 
  1. interface BatchQueue extends Queue {
  2.         void setBatchSize(int batchSize);
  3.         void deleteAll() default {
  4.                 setBatchSize(100);
  5.                 Queue.super.deleteAll();
  6.         }
  7. }

或者子接口也可以通過重新聲明一個沒有方法體的方法來刪除默認的方法:

 
 
 
  1. interface FastQueue extends Queue {
  2.         void deleteAll();
  3. }

這個將強制所有實現(xiàn)了 FastQueue 的類必須實現(xiàn) deleteAll() 方法。

HotSpot 實現(xiàn)

Lambda 不只是可以減少很多代碼的編寫,其字節(jié)碼和運行時的實現(xiàn)也比 Java 7 中的匿名類的效率更高。針對每一個 Lambda 表達式,編譯器都會創(chuàng)建一個對應(yīng)的形如 Lambda$1() 這樣的方法。這個過程被稱之為 Lambda body desugaring. 當(dāng)遇見一個 Lambda 表達式,編譯器將會發(fā)起一個 invokedynamic 調(diào)用,并從目標功能接口中獲取返回值。

深入閱讀

本文很多內(nèi)容都基于 Brian Goetz 的文章:State of the Lambda, State of the Lambda: Libraries Edition and Translation of Lambda Expressions. 這些文字詳細描述了 Lambda 語法、變量捕獲、類型接口和編譯等內(nèi)容。

英文原文:http://datumedge.blogspot.co.uk/2012/06/java-8-Lambdas.html

本文來自:http://www.oschina.net/question/12_59047


文章題目:Java8的Lambda表達式
地址分享:
http://www.dlmjj.cn/article/djgchpj.html