Java8-lambda表達(dá)式.md

Lambda 表達(dá)式詳解

函數(shù)式編程

函數(shù)式接口(functional interface 也叫功能性接口馁启,其實是同一個東西)。簡單來說芍秆,函數(shù)式接口是只包含一個方法的接口惯疙。比如Java標(biāo)準(zhǔn)庫中的java.lang.Runnable和 java.util.Comparator都是典型的函數(shù)式接口。java 8提供 @FunctionalInterface作為注解,這個注解是非必須的妖啥,只要接口符合函數(shù)式接口的標(biāo)準(zhǔn)(即只包含一個方法的接口)霉颠,虛擬機(jī)會自動判斷,但最好在接口上使用注解@FunctionalInterface進(jìn)行聲明荆虱,以免團(tuán)隊的其他人員錯誤地往接口中添加新的方法蒿偎。
Java中的lambda無法單獨出現(xiàn),它需要一個函數(shù)式接口來盛放怀读,lambda表達(dá)式方法體其實就是函數(shù)接口的實現(xiàn)诉位。

Lambda語法

包含三個部分

  • 一個括號內(nèi)用逗號分隔的形式參數(shù),參數(shù)是函數(shù)式接口里面方法的參數(shù)
  • 一個箭頭符號:->
  • 方法體菜枷,可以是表達(dá)式和代碼塊苍糠,方法體函數(shù)式接口里面方法的實現(xiàn),如果是代碼塊啤誊,則必須用{}來包裹起來岳瞭,且需要一個return 返回值,但有個例外蚊锹,若函數(shù)式接口里面方法返回值是void寝优,則無需{}
(parameters) -> expression
(parameters) -> { statements; }

例子如下:

public class TestLambda {

    public static void runThreadUseLambda() {
        //Runnable是一個函數(shù)接口,只包含了有個無參數(shù)的枫耳,返回void的run方法乏矾;
        //所以lambda表達(dá)式左邊沒有參數(shù),右邊也沒有return迁杨,只是單純的打印一句話
        new Thread(() ->System.out.println("lambda實現(xiàn)的線程")).start();
    }

    public static void runThreadUseInnerClass() {
        //這種方式就不多講了钻心,以前舊版本比較常見的做法
        new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("內(nèi)部類實現(xiàn)的線程");
            }
        }).start();
    }

    public static void main(String[] args) {
        TestLambda.runThreadUseLambda();
        TestLambda.runThreadUseInnerClass();
    }
}

可以看出,使用lambda表達(dá)式設(shè)計的代碼會更加簡潔铅协,而且還可讀.

方法引用

其實是lambda表達(dá)式的一個簡化寫法捷沸,所引用的方法其實是lambda表達(dá)式的方法體實現(xiàn),語法也很簡單狐史,左邊是容器(可以是類名痒给,實例名)说墨,中間是"::",右邊是相應(yīng)的方法名苍柏。如下所示:

bjectReference::methodName

一般方法的引用格式是

  • 如果是靜態(tài)方法尼斧,則是ClassName::methodName。如 Object ::equals
  • 如果是實例方法试吁,則是Instance::methodName棺棵。如Object obj=new Object();obj::equals;
  • 構(gòu)造函數(shù).則是ClassName::new

例子如下:

public class TestMethodReference {

    public static void main(String[] args) {

        JFrame frame = new JFrame();
        frame.setLayout(new FlowLayout());
        frame.setVisible(true);

        JButton button1 = new JButton("點我!");
        JButton button2 = new JButton("也點我!");

        frame.getContentPane().add(button1);
        frame.getContentPane().add(button2);
        //這里addActionListener方法的參數(shù)是ActionListener,是一個函數(shù)式接口
        //使用lambda表達(dá)式方式
        button1.addActionListener(e -> { System.out.println("這里是Lambda實現(xiàn)方式"); });
        //使用方法引用方式
        button2.addActionListener(TestMethodReference::doSomething);

    }
    /**
     * 這里是函數(shù)式接口ActionListener的實現(xiàn)方法
     * @param e
     */
    public static void doSomething(ActionEvent e) {

        System.out.println("這里是方法引用實現(xiàn)方式");

    }
}

可以看出熄捍,doSomething方法就是lambda表達(dá)式的實現(xiàn)烛恤,這樣的好處就是,如果你覺得lambda的方法體會很長余耽,影響代碼可讀性缚柏,方法引用就是個解決辦法。

lambda表達(dá)式使用實例

1. lambda表達(dá)式實現(xiàn)Runnable

// Java 8之前:
new Thread(new Runnable() {
  @Override
  public void run() {
    System.out.println("Before Java8, too much code for too little to do");
  }
}).start();
//Java 8方式:
new Thread( () -> System.out.println("In Java8, Lambda expression rocks !!") ).start();

使用lambda表達(dá)式代替匿名類碟贾,可以使代碼更簡短船惨、清晰易于閱讀。

2. lambda表達(dá)式進(jìn)行事件處理

// Java 8之前:
JButton show =  new JButton("Show");
show.addActionListener(new ActionListener() {
    @Override
    public void actionPerformed(ActionEvent e) {
    System.out.println("Event handling without lambda expression is boring");
    }
});

// Java 8方式:
show.addActionListener((e) -> {
    System.out.println("Light, Camera, Action !! Lambda expressions Rocks");
});

3. lambda表達(dá)式對列表進(jìn)行迭代

// Java 8之前:
List features = Arrays.asList("Lambdas", "Default Method", "Stream API", "Date and Time API");
for (String feature : features) {
    System.out.println(feature);
}
// Java 8之后:
List features = Arrays.asList("Lambdas", "Default Method", "Stream API", "Date and Time API");
features.forEach(n -> System.out.println(n));

// 使用Java 8的方法引用更方便缕陕,方法引用由::雙冒號操作符標(biāo)示粱锐,
// 看起來像C++的作用域解析運算符
features.forEach(System.out::println);

由于 Java是命令式語言,Java 8之前的所有循環(huán)代碼都是順序的扛邑,即可以對其元素進(jìn)行并行化處理怜浅。如果你想做并行過濾,就需要自己寫代碼蔬崩,這并不是那么容易恶座。通過引入lambda表達(dá)式 和默認(rèn)方法,將做什么和怎么做的問題分開了沥阳,這意味著Java集合現(xiàn)在知道怎樣做迭代跨琳,并可以在API層面對集合元素進(jìn)行并行處理。

4. lambda表達(dá)式和函數(shù)式接口Predicate

public static void main(String [] args){
    List<String> languages = Arrays.asList("Java", "Scala", "C++", "Haskell", "Lisp");

    System.out.println("Languages which starts with J :");
    filter(languages, (str)-> str.startsWith("J"));

    System.out.println("Languages which ends with a ");
    filter(languages, (str)->str.endsWith("a"));

    System.out.println("Print all languages :");
    filter(languages, (str)->true);

    System.out.println("Print no language : ");
    filter(languages, (str)->false);

    System.out.println("Print language whose length greater than 4:");
    filter(languages, (str)->str.length() > 4);
}

public static void filter(List<String> names, Predicate<String> condition) {
    for(String name: names)  {
        if(condition.test(name)) {
            System.out.println(name + " ");
        }
    }
}
// 更好的辦法
public static void batterFilter(List names, Predicate condition) {
    names.stream().filter((name) -> (condition.test(name))).forEach((name) -> {
        System.out.println(name + " ");
    });
}

除了在語言層面支持函數(shù)式編程風(fēng)格桐罕,Java 8也添加了一個包脉让,叫做 java.util.function。它包含了很多類功炮,用來支持Java的函數(shù)式編程溅潜。其中一個便是Predicate,使用 java.util.function.Predicate 函數(shù)式接口以及l(fā)ambda表達(dá)式薪伏,可以向API方法添加邏輯滚澜,用更少的代碼支持更多的動態(tài)行為

5. 在lambda表達(dá)式中加入Predicate

java.util.function.Predicate 允許將兩個或更多的 Predicate 合成一個。它提供類似于邏輯操作符AND和OR的方法嫁怀,名字叫做and()设捐、or()和xor()借浊,用于將傳入 filter() 方法的條件合并起來。例如萝招,要得到所有以J開始蚂斤,長度為四個字母的語言,可以定義兩個獨立的 Predicate 示例分別表示每一個條件即寒,然后用 Predicate.and() 方法將它們合并起來橡淆,如下所示:

/ 甚至可以用and()召噩、or()和xor()邏輯函數(shù)來合并Predicate母赵,
// 例如要找到所有以J開始,長度為四個字母的名字具滴,你可以合并兩個Predicate并傳入
Predicate<String> startsWithJ = (n) -> n.startsWith("J");
Predicate<String> fourLetterLong = (n) -> n.length() == 4;
names.stream()
    .filter(startsWithJ.and(fourLetterLong))
    .forEach((n) ->
    System.out.print("nName, which starts with 'J' and four letter long is : " + n));

6. lambda表達(dá)式的Map和Reduce示例

// 不使用lambda表達(dá)式為每個訂單加上12%的稅
List<Integer> costBeforeTax = Arrays.asList(100, 200, 300, 400, 500);
for (Integer cost : costBeforeTax) {
    double price = cost + .12*cost;
    System.out.println(price);
}
// 使用lambda表達(dá)式
costBeforeTax.stream().map((cost) -> cost + .12*cost).forEach(System.out::println);

在上個例子中凹嘲,可以看到map將集合類(例如列表)元素進(jìn)行轉(zhuǎn)換的。還有一個 reduce() 函數(shù)可以將所有值合并成一個构韵。Map和Reduce操作是函數(shù)式編程的核心操作周蹭,因為其功能,reduce 又被稱為折疊操作疲恢。另外凶朗,reduce 并不是一個新的操作,你有可能已經(jīng)在使用它显拳。SQL中類似 sum()棚愤、avg() 或者 count() 的聚集函數(shù),實際上就是 reduce 操作杂数,因為它們接收多個值并返回一個值宛畦。流API定義的 reduceh() 函數(shù)可以接受lambda表達(dá)式,并對所有值進(jìn)行合并揍移。IntStream這樣的類有類似 average()次和、count()、sum() 的內(nèi)建方法來做 reduce 操作那伐,也有mapToLong()踏施、mapToDouble() 方法來做轉(zhuǎn)換。這并不會限制你罕邀,你可以用內(nèi)建方法读规,也可以自己定義。在這個Java 8的Map Reduce示例里燃少,我們首先對所有價格應(yīng)用 12% 的VAT束亏,然后用 reduce() 方法計算總和。

// 為每個訂單加上12%的稅
// 老方法:
List costBeforeTax = Arrays.asList(100, 200, 300, 400, 500);
double total = 0;
for (Integer cost : costBeforeTax) {
    double price = cost + .12*cost;
    total = total + price;
}
System.out.println("Total : " + total);

// 新方法:
List costBeforeTax = Arrays.asList(100, 200, 300, 400, 500);
double bill = costBeforeTax.stream().map((cost) -> cost + .12*cost).reduce((sum, cost) -> sum + cost).get();
System.out.println("Total : " + bill);

7. 通過過濾創(chuàng)建一個String列表

// 創(chuàng)建一個字符串列表阵具,每個字符串長度大于2
List<String> filtered = strList.stream().filter(x -> x.length()> 2).collect(Collectors.toList());
System.out.printf("Original List : %s, filtered list : %s %n", strList, filtered);

過濾是Java開發(fā)者在大規(guī)模集合上的一個常用操作碍遍,而現(xiàn)在使用lambda表達(dá)式和流API過濾大規(guī)模數(shù)據(jù)集合是驚人的簡單定铜。流提供了一個 filter() 方法,接受一個 Predicate 對象怕敬,即可以傳入一個lambda表達(dá)式作為過濾邏輯

8. 對列表的每個元素應(yīng)用函數(shù)

// 將字符串換成大寫并用逗號鏈接起來
List<String> G7 = Arrays.asList("USA", "Japan", "France", "Germany", "Italy", "U.K.","Canada");
String G7Countries = G7.stream().map(x -> x.toUpperCase()).collect(Collectors.joining(", "));
System.out.println(G7Countries);

我們通常需要對列表的每個元素使用某個函數(shù)揣炕,例如逐一乘以某個數(shù)、除以某個數(shù)或者做其它操作东跪。這些操作都很適合用 map() 方法畸陡,可以將轉(zhuǎn)換邏輯以lambda表達(dá)式的形式放在 map() 方法里,就可以對集合的各個元素進(jìn)行轉(zhuǎn)換了

9. 復(fù)制不同的值虽填,創(chuàng)建一個子列表

// 用所有不同的數(shù)字創(chuàng)建一個正方形列表
List<Integer> numbers = Arrays.asList(9, 10, 3, 4, 7, 3, 4);
List<Integer> distinct = numbers.stream().map( i -> i*i).distinct().collect(Collectors.toList());
System.out.printf("Original List : %s,  Square Without duplicates : %s %n", numbers, distinct);

利用流的 distinct() 方法來對集合進(jìn)行去重

10. 計算集合元素的最大值丁恭、最小值、總和以及平均值

//獲取數(shù)字的個數(shù)斋日、最小值牲览、最大值、總和以及平均值
List<Integer> primes = Arrays.asList(2, 3, 5, 7, 11, 13, 17, 19, 23, 29);
IntSummaryStatistics stats = primes.stream().mapToInt((x) -> x).summaryStatistics();
System.out.println("Highest prime number in List : " + stats.getMax());
System.out.println("Lowest prime number in List : " + stats.getMin());
System.out.println("Sum of all prime numbers : " + stats.getSum());
System.out.println("Average of all prime numbers : " + stats.getAverage());

IntStream恶守、LongStream 和 DoubleStream 等流的類中第献,有個非常有用的方法叫做 summaryStatistics() ⊥酶郏可以返回 IntSummaryStatistics庸毫、LongSummaryStatistics 或者 DoubleSummaryStatistic s,描述流中元素的各種摘要數(shù)據(jù)衫樊。在本例中飒赃,我們用這個方法來計算列表的最大值和最小值。它也有 getSum() 和 getAverage() 方法來獲得列表的所有元素的總和及平均值

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末橡伞,一起剝皮案震驚了整個濱河市盒揉,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌兑徘,老刑警劉巖刚盈,帶你破解...
    沈念sama閱讀 217,734評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異挂脑,居然都是意外死亡藕漱,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,931評論 3 394
  • 文/潘曉璐 我一進(jìn)店門崭闲,熙熙樓的掌柜王于貴愁眉苦臉地迎上來肋联,“玉大人,你說我怎么就攤上這事刁俭¢先裕” “怎么了?”我有些...
    開封第一講書人閱讀 164,133評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長侮繁。 經(jīng)常有香客問我虑粥,道長,這世上最難降的妖魔是什么宪哩? 我笑而不...
    開封第一講書人閱讀 58,532評論 1 293
  • 正文 為了忘掉前任娩贷,我火速辦了婚禮,結(jié)果婚禮上锁孟,老公的妹妹穿的比我還像新娘彬祖。我一直安慰自己,他們只是感情好品抽,可當(dāng)我...
    茶點故事閱讀 67,585評論 6 392
  • 文/花漫 我一把揭開白布储笑。 她就那樣靜靜地躺著,像睡著了一般桑包。 火紅的嫁衣襯著肌膚如雪南蓬。 梳的紋絲不亂的頭發(fā)上纺非,一...
    開封第一講書人閱讀 51,462評論 1 302
  • 那天哑了,我揣著相機(jī)與錄音,去河邊找鬼烧颖。 笑死弱左,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的炕淮。 我是一名探鬼主播拆火,決...
    沈念sama閱讀 40,262評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼涂圆!你這毒婦竟也來了们镜?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,153評論 0 276
  • 序言:老撾萬榮一對情侶失蹤润歉,失蹤者是張志新(化名)和其女友劉穎模狭,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體踩衩,經(jīng)...
    沈念sama閱讀 45,587評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡嚼鹉,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,792評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了驱富。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片锚赤。...
    茶點故事閱讀 39,919評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖褐鸥,靈堂內(nèi)的尸體忽然破棺而出线脚,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 35,635評論 5 345
  • 正文 年R本政府宣布浑侥,位于F島的核電站又憨,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏锭吨。R本人自食惡果不足惜蠢莺,卻給世界環(huán)境...
    茶點故事閱讀 41,237評論 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望零如。 院中可真熱鬧躏将,春花似錦、人聲如沸考蕾。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,855評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽肖卧。三九已至蚯窥,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間塞帐,已是汗流浹背拦赠。 一陣腳步聲響...
    開封第一講書人閱讀 32,983評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留葵姥,地道東北人荷鼠。 一個月前我還...
    沈念sama閱讀 48,048評論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像榔幸,于是被迫代替她去往敵國和親允乐。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,864評論 2 354

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