Java8的Lambda特性

oracle官網(wǎng)教程

  • 概述
  • 背景
  • 例子
  • 使用Lambda表達(dá)式改進代碼
  • 總結(jié)

概述

Lambda表達(dá)式是Java8的一個重要的新特性港华,它使用一個表達(dá)式來表示一個方法接口塘偎。它還改進了Collection庫航唆,新的并發(fā)特性也提升了在多核環(huán)境下的性能。

背景

匿名內(nèi)部類

首先還要先從匿名內(nèi)部類說起,如果一個類只出現(xiàn)一次,那么可以使用如下方式實現(xiàn):

 JButton testButton = new JButton("Test Button");
 testButton.addActionListener(new ActionListener(){
     @Override public void actionPerformed(ActionEvent ae){
         System.out.println("Click Detected by Anon Class");
     }
 });

上面這段代碼可讀性不是很好赶站,也不夠優(yōu)雅。

功能性接口

ActionListener接口

package java.awt.event;
import java.util.EventListener;

public interface ActionListener extends EventListener {
    public void actionPerformed(ActionEvent e);
}

像ActionListener只有一個方法的接口叫功能性接口吓揪,比如 Runnable 和 Comparator.

Lambda表達(dá)式語法

一個Lambda表達(dá)式由以下三部分組成:

Argument list Arrow token Body
(int x, int y) -> x+y

body可以是一個表達(dá)式或者一個語句塊亲怠。如果是表達(dá)式的形式,就執(zhí)行body的代碼并返回柠辞;如果是代碼塊的形式团秽,執(zhí)行完之后會把控制權(quán)交給調(diào)用方。

代碼塊的形式:

(String s) -> { System.out.println(s); }

表達(dá)式的形式:

(int x, int y) -> x + y

() -> 42

例子

下面是Comparator的例子叭首,使用了內(nèi)部類和lambda表達(dá)式习勤。

public class Person {
   private String givenName;
   private String surName;
   private int age;
   private Gender gender;
   private String eMail;
   private String phone;
   private String address;

   /** getter , setter and other methods **/
}
     List<Person> personList = Person.createShortList();
   
     // Sort with Inner Class
     Collections.sort(personList, new Comparator<Person>(){
       public int compare(Person p1, Person p2){
         return p1.getSurName().compareTo(p2.getSurName());
       }
     });
     System.out.println("=== Sorted Asc SurName ===");
     for(Person p:personList){
       p.printName();
     }
     
     // Use Lambda instead     
     // Print Asc
     System.out.println("=== Sorted Asc SurName ===");
     Collections.sort(personList, (Person p1, Person p2) -> p1.getSurName().compareTo(p2.getSurName()));
     for(Person p:personList){
       p.printName();
     }     

     // Print Desc
     System.out.println("=== Sorted Desc SurName ===");
     Collections.sort(personList, (p1,  p2) -> p2.getSurName().compareTo(p1.getSurName()));
     for(Person p:personList){
       p.printName();
     }

可以注意到第二個lambda表達(dá)式?jīng)]有聲明參數(shù)類型,lambda支持“target typing”焙格,從上下文推斷出參數(shù)類型图毕。這里因為Comparator定義的是泛型,編譯器可以推斷出這里的兩個參數(shù)類型是Person眷唉。

使用Lambda表達(dá)式改進代碼

理論上lambda應(yīng)該支持DRY(Don't repeat yourself)原則予颤,并且使代碼可讀性更好。
下面這個例子依舊使用上面的Person類冬阳。假設(shè)我們需要區(qū)分一個集合里的Person的身份:

  • Drivers:age>=16
  • Draftees: age>=18 && ages<=25的男性青年
  • Pilots: age>=23 && ages<=65

下面這個類負(fù)責(zé)查找集合中的Driver蛤虐、Draftee和Pilot

public class RoboContactMethods {   
     public void callDrivers(List<Person> pl){
       for(Person p:pl){
         if (p.getAge() >= 16){
           roboCall(p);
         }
       }
     }
    
     public void emailDraftees(List<Person> pl){
       for(Person p:pl){
         if (p.getAge() >= 18 && p.getAge() <= 25 && p.getGender() == Gender.MALE){
           roboEmail(p);
         }
       }
     }
     
     public void mailPilots(List<Person> pl){
       for(Person p:pl){
         if (p.getAge() >= 23 && p.getAge() <= 65){
           roboMail(p);
         }
       }
     }      

    public void roboCall(Person p){
       System.out.println("Calling " + p.getGivenName() + " " + p.getSurName() + " age " + p.getAge() + " at " + p.getPhone());
     }
     
     public void roboEmail(Person p){
       System.out.println("EMailing " + p.getGivenName() + " " + p.getSurName() + " age " + p.getAge() + " at " + p.getEmail());
     }
     
     public void roboMail(Person p){
       System.out.println("Mailing " + p.getGivenName() + " " + p.getSurName() + " age " + p.getAge() + " at " + p.getAddress());
     }
   }

很明顯上面這個類不符合DRY原則,每個方法都使用了循環(huán)肝陪,代碼也不易維護驳庭。如果改變了查找的規(guī)則,那這段代碼的改動會很大。
重構(gòu)
Java SE 8提供了java.util.function包饲常,有很多標(biāo)準(zhǔn)的功能性接口蹲堂,比如Predicate:

public interface Predicate<T> {
   public boolean test(T t);
 }

下面是使用Predicate接口改進的RoboContactMethods類:

public class RoboContactLambda {
     public void phoneContacts(List<Person> pl, Predicate<Person> pred){
       for(Person p:pl){
         if (pred.test(p)){
           roboCall(p);
         }
       }
     }
   
     public void emailContacts(List<Person> pl, Predicate<Person> pred){
       for(Person p:pl){
         if (pred.test(p)){
           roboEmail(p);
         }
       }
     }
   
     public void mailContacts(List<Person> pl, Predicate<Person> pred){
       for(Person p:pl){
         if (pred.test(p)){
           roboMail(p);
         }
       }
     }  
     
     public void roboCall(Person p){
       System.out.println("Calling " + p.getGivenName() + " " + p.getSurName() + " age " + p.getAge() + " at " + p.getPhone());
     }
     
     public void roboEmail(Person p){
       System.out.println("EMailing " + p.getGivenName() + " " + p.getSurName() + " age " + p.getAge() + " at " + p.getEmail());
     }
     
     public void roboMail(Person p){
       System.out.println("Mailing " + p.getGivenName() + " " + p.getSurName() + " age " + p.getAge() + " at " + p.getAddress());
     }
   }

調(diào)用:

       List<Person> pl = Person.createShortList();
       RoboContactLambda robo = new RoboContactLambda();
       
       // Predicates
       Predicate<Person> allDrivers = p -> p.getAge() >= 16;
       Predicate<Person> allDraftees = p -> p.getAge() >= 18 && p.getAge() <= 25 && p.getGender() == Gender.MALE;
       Predicate<Person> allPilots = p -> p.getAge() >= 23 && p.getAge() <= 65;
     

總結(jié)

本文主要介紹了Java的匿名內(nèi)部類,使用lambda表達(dá)式代替匿名內(nèi)部類贝淤,lambda表達(dá)式的正確語法柒竞,以及function包中Predicate接口。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末霹娄,一起剝皮案震驚了整個濱河市能犯,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌犬耻,老刑警劉巖踩晶,帶你破解...
    沈念sama閱讀 218,036評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異枕磁,居然都是意外死亡渡蜻,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,046評論 3 395
  • 文/潘曉璐 我一進店門计济,熙熙樓的掌柜王于貴愁眉苦臉地迎上來茸苇,“玉大人,你說我怎么就攤上這事沦寂⊙埽” “怎么了?”我有些...
    開封第一講書人閱讀 164,411評論 0 354
  • 文/不壞的土叔 我叫張陵传藏,是天一觀的道長腻暮。 經(jīng)常有香客問我,道長毯侦,這世上最難降的妖魔是什么哭靖? 我笑而不...
    開封第一講書人閱讀 58,622評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮侈离,結(jié)果婚禮上试幽,老公的妹妹穿的比我還像新娘。我一直安慰自己卦碾,他們只是感情好铺坞,可當(dāng)我...
    茶點故事閱讀 67,661評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著洲胖,像睡著了一般康震。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上宾濒,一...
    開封第一講書人閱讀 51,521評論 1 304
  • 那天,我揣著相機與錄音屏箍,去河邊找鬼绘梦。 笑死橘忱,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的卸奉。 我是一名探鬼主播钝诚,決...
    沈念sama閱讀 40,288評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼榄棵!你這毒婦竟也來了凝颇?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,200評論 0 276
  • 序言:老撾萬榮一對情侶失蹤疹鳄,失蹤者是張志新(化名)和其女友劉穎拧略,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體瘪弓,經(jīng)...
    沈念sama閱讀 45,644評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡垫蛆,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,837評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了腺怯。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片袱饭。...
    茶點故事閱讀 39,953評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖呛占,靈堂內(nèi)的尸體忽然破棺而出虑乖,到底是詐尸還是另有隱情,我是刑警寧澤晾虑,帶...
    沈念sama閱讀 35,673評論 5 346
  • 正文 年R本政府宣布疹味,位于F島的核電站,受9級特大地震影響走贪,放射性物質(zhì)發(fā)生泄漏佛猛。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,281評論 3 329
  • 文/蒙蒙 一坠狡、第九天 我趴在偏房一處隱蔽的房頂上張望继找。 院中可真熱鬧,春花似錦逃沿、人聲如沸婴渡。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,889評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽边臼。三九已至,卻和暖如春假消,著一層夾襖步出監(jiān)牢的瞬間柠并,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,011評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留臼予,地道東北人鸣戴。 一個月前我還...
    沈念sama閱讀 48,119評論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像粘拾,于是被迫代替她去往敵國和親窄锅。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,901評論 2 355

推薦閱讀更多精彩內(nèi)容