Java 8 lambda 表達(dá)式10個(gè)示例

Java 8 lambda 表達(dá)式10個(gè)示例

Java 8 發(fā)布于4年前抖部,日期是2014年3月18日说贝,這次開創(chuàng)性的發(fā)布在Java社區(qū)引發(fā)了不少討論,并讓大家感到激動(dòng)慎颗。特性之一便是隨同發(fā)布的lambda表達(dá)式乡恕,它將允許我們將行為傳到函數(shù)里。在Java 8之前俯萎,如果想將行為傳入函數(shù)傲宜,僅有的選擇就是匿名類,需要6行代碼夫啊。而定義行為最重要的那行代碼函卒,卻混在中間不夠突出。Lambda表達(dá)式取代了匿名類撇眯,取消了模板报嵌,允許用函數(shù)式風(fēng)格編寫代碼。這樣有時(shí)可讀性更好熊榛,表達(dá)更清晰锚国。

在Java生態(tài)系統(tǒng)中,函數(shù)式表達(dá)與對(duì)面向?qū)ο蟮娜嬷С质莻€(gè)激動(dòng)人心的進(jìn)步玄坦。將進(jìn)一步促進(jìn)并行第三方庫(kù)的發(fā)展血筑,充分利用多核CPU。

盡管業(yè)界需要時(shí)間來(lái)消化Java 8,但我認(rèn)為任何嚴(yán)謹(jǐn)?shù)腏ava開發(fā)者都不應(yīng)忽視此次Java發(fā)布的核心特性豺总,即lambda表達(dá)式车伞、函數(shù)式接口、流API园欣、默認(rèn)方法和新的Date以及Time API帖世。

作為開發(fā)人員,我發(fā)現(xiàn)學(xué)習(xí)和掌握l(shuí)ambda表達(dá)式的最佳方法就是勇于嘗試沸枯,盡可能多練習(xí)lambda表達(dá)式例子日矫。鑒于受Java 8發(fā)布的影響最大的是Java集合框架(Java Collections framework),所以最好練習(xí)流API和lambda表達(dá)式绑榴,用于對(duì)列表(Lists)和集合(Collections)數(shù)據(jù)進(jìn)行提取哪轿、過(guò)濾和排序。

我一直在進(jìn)行關(guān)于Java 8的寫作翔怎,過(guò)去也曾分享過(guò)一些資源來(lái)幫助大家掌握J(rèn)ava 8窃诉。本文分享在代碼中最有用的10個(gè)lambda表達(dá)式的使用方法,這些例子都短小精悍赤套,將幫助你快速學(xué)會(huì)lambda表達(dá)式飘痛。

Java 8 lambda表達(dá)式示例

我個(gè)人對(duì)Java 8發(fā)布非常激動(dòng),尤其是lambda表達(dá)式和流API容握。越來(lái)越多的了解它們宣脉,我能寫出更干凈的代碼。雖然一開始并不是這樣剔氏。第一次看到用lambda表達(dá)式寫出來(lái)的Java代碼時(shí)塑猖,我對(duì)這種神秘的語(yǔ)法感到非常失望,認(rèn)為它們把Java搞得不可讀谈跛,但我錯(cuò)了羊苟。花了一天時(shí)間做了一些lambda表達(dá)式和流API示例的練習(xí)后感憾,我開心的看到了更清晰的Java代碼蜡励。這有點(diǎn)像學(xué)習(xí)泛型,第一次見(jiàn)的時(shí)候我很討厭它吹菱。我甚至繼續(xù)使用老版Java 1.4來(lái)處理集合巍虫,直到有一天,朋友跟我介紹了使用泛型的好處(才意識(shí)到它的好處)鳍刷。所以基本立場(chǎng)就是占遥,不要畏懼lambda表達(dá)式以及方法引用的神秘語(yǔ)法,做幾次練習(xí)输瓜,從集合類中提取瓦胎、過(guò)濾數(shù)據(jù)之后芬萍,你就會(huì)喜歡上它。下面讓我們開啟學(xué)習(xí)Java 8 lambda表達(dá)式的學(xué)習(xí)之旅吧搔啊,首先從簡(jiǎn)單例子開始柬祠。

例1、用lambda表達(dá)式實(shí)現(xiàn)Runnable

我開始使用Java 8時(shí)负芋,首先做的就是使用lambda表達(dá)式替換匿名類漫蛔,而實(shí)現(xiàn)Runnable接口是匿名類的最好示例【啥辏看一下Java 8之前的runnable實(shí)現(xiàn)方法莽龟,需要4行代碼,而使用lambda表達(dá)式只需要一行代碼锨天。我們?cè)谶@里做了什么呢毯盈?那就是用() -> {}代碼塊替代了整個(gè)匿名類

// Java 8之前:new Thread(newRunnable() {? ? @Override? ? public voidrun() {? ? 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();

輸出:

too much code,fortoo little todoLambda expression rocks !!

這個(gè)例子向我們展示了Java 8 lambda表達(dá)式的語(yǔ)法病袄。你可以使用lambda寫出如下代碼:

(params) -> expression

(params) -> statement

(params) -> { statements }

例如搂赋,如果你的方法不對(duì)參數(shù)進(jìn)行修改、重寫益缠,只是在控制臺(tái)打印點(diǎn)東西的話脑奠,那么可以這樣寫:

() -> System.out.println("Hello Lambda Expressions");

如果你的方法接收兩個(gè)參數(shù),那么可以寫成如下這樣:

(int even, int odd) -> even + odd

順便提一句幅慌,通常都會(huì)把lambda表達(dá)式內(nèi)部變量的名字起得短一些捺信。這樣能使代碼更簡(jiǎn)短,放在同一行欠痴。所以,在上述代碼中秒咨,變量名選用a喇辽、b或者x、y會(huì)比even雨席、odd要好菩咨。

例2、使用Java 8 lambda表達(dá)式進(jìn)行事件處理

如果你用過(guò)Swing API編程陡厘,你就會(huì)記得怎樣寫事件監(jiān)聽(tīng)代碼抽米。這又是一個(gè)舊版本簡(jiǎn)單匿名類的經(jīng)典用例,但現(xiàn)在可以不這樣了糙置。你可以用lambda表達(dá)式寫出更好的事件監(jiān)聽(tīng)代碼云茸,如下所示:

// Java 8之前:JButton show =? new JButton("Show");show.addActionListener(newActionListener() {? ? @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");});

Java開發(fā)者經(jīng)常使用匿名類的另一個(gè)地方是為?Collections.sort()?定制?Comparator。在Java 8中谤饭,你可以用更可讀的lambda表達(dá)式換掉丑陋的匿名類标捺。我把這個(gè)留做練習(xí)懊纳,應(yīng)該不難,可以按照我在使用lambda表達(dá)式實(shí)現(xiàn)?Runnable?和?ActionListener?的過(guò)程中的套路來(lái)做亡容。

例3嗤疯、使用lambda表達(dá)式對(duì)列表進(jìn)行迭代

如果你使過(guò)幾年Java,你就知道針對(duì)集合類闺兢,最常見(jiàn)的操作就是進(jìn)行迭代茂缚,并將業(yè)務(wù)邏輯應(yīng)用于各個(gè)元素,例如處理訂單屋谭、交易和事件的列表脚囊。由于Java是命令式語(yǔ)言,Java 8之前的所有循環(huán)代碼都是順序的戴而,即可以對(duì)其元素進(jìn)行并行化處理凑术。如果你想做并行過(guò)濾,就需要自己寫代碼所意,這并不是那么容易淮逊。通過(guò)引入lambda表達(dá)式和默認(rèn)方法,將做什么和怎么做的問(wèn)題分開了扶踊,這意味著Java集合現(xiàn)在知道怎樣做迭代泄鹏,并可以在API層面對(duì)集合元素進(jìn)行并行處理。下面的例子里秧耗,我將介紹如何在使用lambda或不使用lambda表達(dá)式的情況下迭代列表备籽。你可以看到列表現(xiàn)在有了一個(gè) forEach() 方法,它可以迭代所有對(duì)象分井,并將你的lambda代碼應(yīng)用在其中车猬。

// 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的方法引用更方便,方法引用由::雙冒號(hào)操作符標(biāo)示尺锚,// 看起來(lái)像C++的作用域解析運(yùn)算符features.forEach(System.out::println);

輸出:

Lambdas

Default Method

Stream API

Date and Time API

列表循環(huán)的最后一個(gè)例子展示了如何在Java 8中使用方法引用(method reference)珠闰。你可以看到C++里面的雙冒號(hào)、范圍解析操作符現(xiàn)在在Java 8中用來(lái)表示方法引用瘫辩。

例4伏嗜、使用lambda表達(dá)式和函數(shù)式接口Predicate

除了在語(yǔ)言層面支持函數(shù)式編程風(fēng)格,Java 8也添加了一個(gè)包伐厌,叫做?java.util.function承绸。它包含了很多類,用來(lái)支持Java的函數(shù)式編程挣轨。其中一個(gè)便是Predicate军熏,使用java.util.function.Predicate?函數(shù)式接口以及l(fā)ambda表達(dá)式,可以向API方法添加邏輯刃唐,用更少的代碼支持更多的動(dòng)態(tài)行為羞迷。下面是Java 8 Predicate 的例子界轩,展示了過(guò)濾集合數(shù)據(jù)的多種常用方法。Predicate接口非常適用于做過(guò)濾衔瓮。

public static void main(String[] args) {? ? List languages = Arrays.asList("Java","Scala","C++","Haskell","Lisp");? ? System.out.println("Languages which starts with J :");? ? filter(languages, (str)->((String)str).startsWith("J"));? ? System.out.println("Languages which ends with a ");? ? filter(languages, (str)->((String)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)->((String)str).length() > 4);}public static void filter(List names, Predicate condition) {for(String? name: names)? {if(condition.test(name)) {? ? ? ? ? ? System.out.println(name +" ");? ? ? ? }? ? }}

輸出:

Languageswhichstarts with J :JavaLanguageswhichends with aJavaScalaPrint all languages :JavaScalaC++HaskellLispPrint no language :Print language whose length greater than 4:ScalaHaskell// 更好的辦法public static void filter(List names, Predicate condition) {? ? names.stream().filter((name) -> (condition.test(name))).forEach((name) -> {? ? ? ? System.out.println(name +" ");? ? });}

可以看到浊猾,Stream API的過(guò)濾方法也接受一個(gè)Predicate,這意味著可以將我們定制的 filter() 方法替換成寫在里面的內(nèi)聯(lián)代碼热鞍,這就是lambda表達(dá)式的魔力葫慎。另外,Predicate接口也允許進(jìn)行多重條件的測(cè)試薇宠,下個(gè)例子將要講到偷办。

例5、如何在lambda表達(dá)式中加入Predicate

上個(gè)例子說(shuō)到澄港,java.util.function.Predicate?允許將兩個(gè)或更多的?Predicate?合成一個(gè)椒涯。它提供類似于邏輯操作符AND和OR的方法,名字叫做?and()回梧、or()废岂,用于將傳入?filter()?方法的條件合并起來(lái)。例如狱意,要得到所有以J開始湖苞,長(zhǎng)度為四個(gè)字母的語(yǔ)言,可以定義兩個(gè)獨(dú)立的?Predicate?示例分別表示每一個(gè)條件详囤,然后用?Predicate.and()?方法將它們合并起來(lái)财骨,如下所示

// 甚至可以用and()、or()邏輯函數(shù)來(lái)合并Predicate藏姐,// 例如要找到所有以J開始隆箩,長(zhǎng)度為四個(gè)字母的名字,你可以合并兩個(gè)Predicate并傳入Predicate startsWithJ = (n) -> n.startsWith("J");Predicate 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));

類似地羔杨,也可以使用?or()?方法摘仅。本例著重介紹了如下要點(diǎn):可按需要將?Predicate?作為單獨(dú)條件然后將其合并起來(lái)使用。簡(jiǎn)而言之问畅,你可以以傳統(tǒng)Java命令方式使用?Predicate?接口,也可以充分利用lambda表達(dá)式達(dá)到事半功倍的效果六荒。

例6护姆、Java 8中使用lambda表達(dá)式的Map和Reduce示例

本例介紹最廣為人知的函數(shù)式編程概念map。它允許你將對(duì)象進(jìn)行轉(zhuǎn)換掏击。例如在本例中卵皂,我們將costBeforeTax?列表的每個(gè)元素轉(zhuǎn)換成為稅后的值。我們將?x -> x*x?lambda表達(dá)式傳到 map() 方法砚亭,后者將其應(yīng)用到流中的每一個(gè)元素灯变。然后用 forEach() 將列表元素打印出來(lái)殴玛。使用流API的收集器類,可以得到所有含稅的開銷添祸。有 toList() 這樣的方法將 map 或任何其他操作的結(jié)果合并起來(lái)滚粟。由于收集器在流上做終端操作,因此之后便不能重用流了刃泌。你甚至可以用流API的 reduce() 方法將所有數(shù)字合成一個(gè)凡壤,下一個(gè)例子將會(huì)講到。

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

輸出:

112.0

224.0

336.0

448.0

560.0

112.0

224.0

336.0

448.0

560.0

例6.2耙替、Java 8中使用lambda表達(dá)式的Map和Reduce示例

在上個(gè)例子中亚侠,可以看到map將集合類(例如列表)元素進(jìn)行轉(zhuǎn)換的。還有一個(gè)?reduce()?函數(shù)可以將所有值合并成一個(gè)俗扇。Map和Reduce操作是函數(shù)式編程的核心操作硝烂,因?yàn)槠涔δ埽瑀educe 又被稱為折疊操作铜幽。另外滞谢,reduce 并不是一個(gè)新的操作,你有可能已經(jīng)在使用它啥酱。SQL中類似sum()爹凹、avg()?或者?count()?的聚集函數(shù),實(shí)際上就是 reduce 操作镶殷,因?yàn)樗鼈兘邮斩鄠€(gè)值并返回一個(gè)值禾酱。流API定義的 reduceh() 函數(shù)可以接受lambda表達(dá)式,并對(duì)所有值進(jìn)行合并绘趋。IntStream這樣的類有類似?average()颤陶、count()、sum()?的內(nèi)建方法來(lái)做 reduce 操作陷遮,也有mapToLong()滓走、mapToDouble()?方法來(lái)做轉(zhuǎn)換。這并不會(huì)限制你帽馋,你可以用內(nèi)建方法搅方,也可以自己定義。在這個(gè)Java 8的Map Reduce示例里绽族,我們首先對(duì)所有價(jià)格應(yīng)用?12%?的VAT姨涡,然后用reduce()?方法計(jì)算總和。

// 為每個(gè)訂單加上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);

輸出:

Total : 1680.0

Total : 1680.0

例7吧慢、通過(guò)過(guò)濾創(chuàng)建一個(gè)String列表

過(guò)濾是Java開發(fā)者在大規(guī)模集合上的一個(gè)常用操作涛漂,而現(xiàn)在使用lambda表達(dá)式和流API過(guò)濾大規(guī)模數(shù)據(jù)集合是驚人的簡(jiǎn)單。流提供了一個(gè)?filter()?方法,接受一個(gè)?Predicate?對(duì)象匈仗,即可以傳入一個(gè)lambda表達(dá)式作為過(guò)濾邏輯瓢剿。下面的例子是用lambda表達(dá)式過(guò)濾Java集合,將幫助理解悠轩。

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

輸出:

Original List : [abc, , bcd, , defg, jk], filtered list : [abc, bcd, defg]

另外,關(guān)于 filter() 方法有個(gè)常見(jiàn)誤解哗蜈。在現(xiàn)實(shí)生活中前标,做過(guò)濾的時(shí)候,通常會(huì)丟棄部分距潘,但使用filter()方法則是獲得一個(gè)新的列表炼列,且其每個(gè)元素符合過(guò)濾原則。

例8音比、對(duì)列表的每個(gè)元素應(yīng)用函數(shù)

我們通常需要對(duì)列表的每個(gè)元素使用某個(gè)函數(shù)俭尖,例如逐一乘以某個(gè)數(shù)、除以某個(gè)數(shù)或者做其它操作洞翩。這些操作都很適合用?map()?方法稽犁,可以將轉(zhuǎn)換邏輯以lambda表達(dá)式的形式放在 map() 方法里,就可以對(duì)集合的各個(gè)元素進(jìn)行轉(zhuǎn)換了骚亿,如下所示已亥。

// 將字符串換成大寫并用逗號(hào)鏈接起來(lái)List 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);

輸出:

USA, JAPAN, FRANCE, GERMANY, ITALY, U.K., CANADA

例9、復(fù)制不同的值来屠,創(chuàng)建一個(gè)子列表

本例展示了如何利用流的?distinct()?方法來(lái)對(duì)集合進(jìn)行去重虑椎。

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

輸出:

Original List : [9, 10, 3, 4, 7, 3, 4],? Square Without duplicates : [81, 100, 9, 16, 49]

例10、計(jì)算集合元素的最大值俱笛、最小值捆姜、總和以及平均值

IntStreamLongStream?和?DoubleStream?等流的類中迎膜,有個(gè)非常有用的方法叫做summaryStatistics()?泥技。可以返回?IntSummaryStatistics磕仅、LongSummaryStatistics?或者DoubleSummaryStatistics珊豹,描述流中元素的各種摘要數(shù)據(jù)。在本例中榕订,我們用這個(gè)方法來(lái)計(jì)算列表的最大值和最小值平夜。它也有?getSum()?和?getAverage()?方法來(lái)獲得列表的所有元素的總和及平均值。

//獲取數(shù)字的個(gè)數(shù)卸亮、最小值、最大值玩裙、總和以及平均值List 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());

輸出:

Highest prime numberinList : 29Lowest prime numberinList : 2Sum of all prime numbers : 129Average of all prime numbers : 12.9

Lambda表達(dá)式 vs 匿名類

既然lambda表達(dá)式即將正式取代Java代碼中的匿名內(nèi)部類兼贸,那么有必要對(duì)二者做一個(gè)比較分析段直。一個(gè)關(guān)鍵的不同點(diǎn)就是關(guān)鍵字?this。匿名類的?this?關(guān)鍵字指向匿名類溶诞,而lambda表達(dá)式的?this關(guān)鍵字指向包圍lambda表達(dá)式的類鸯檬。另一個(gè)不同點(diǎn)是二者的編譯方式。Java編譯器將lambda表達(dá)式編譯成類的私有方法螺垢。使用了Java 7的?invokedynamic?字節(jié)碼指令來(lái)動(dòng)態(tài)綁定這個(gè)方法喧务。

Java 8 Lambda表達(dá)式要點(diǎn)

10個(gè)Java lambda表達(dá)式、流API示例

到目前為止我們看到了Java 8的10個(gè)lambda表達(dá)式枉圃,這對(duì)于新手來(lái)說(shuō)是個(gè)合適的任務(wù)量功茴,你可能需要親自運(yùn)行示例程序以便掌握。試著修改要求創(chuàng)建自己的例子孽亲,達(dá)到快速學(xué)習(xí)的目的坎穿。我還想建議大家使用Netbeans IDE來(lái)練習(xí)lambda表達(dá)式,它對(duì)Java 8支持良好返劲。當(dāng)把代碼轉(zhuǎn)換成函數(shù)式的時(shí)候玲昧,Netbeans會(huì)及時(shí)給你提示。只需跟著Netbeans的提示篮绿,就能很容易地把匿名類轉(zhuǎn)換成lambda表達(dá)式孵延。此外,如果你喜歡閱讀亲配,那么記得看一下Java 8的lambdas尘应,實(shí)用函數(shù)式編程這本書(Java 8 Lambdas, pragmatic functional programming),作者是Richard Warburton弃榨,或者也可以看看Manning的Java 8實(shí)戰(zhàn)(Java 8 in Action)菩收,這本書雖然還沒(méi)出版,但我猜線上有第一章的免費(fèi)pdf鲸睛。不過(guò)娜饵,在你開始忙其它事情之前,先回顧一下Java 8的lambda表達(dá)式官辈、默認(rèn)方法和函數(shù)式接口的重點(diǎn)知識(shí)箱舞。

1)lambda表達(dá)式僅能放入如下代碼:預(yù)定義使用了 @Functional 注釋的函數(shù)式接口,自帶一個(gè)抽象函數(shù)的方法拳亿,或者SAM(Single Abstract Method 單個(gè)抽象方法)類型晴股。這些稱為lambda表達(dá)式的目標(biāo)類型,可以用作返回類型肺魁,或lambda目標(biāo)代碼的參數(shù)电湘。例如,若一個(gè)方法接收Runnable、Comparable或者 Callable 接口寂呛,都有單個(gè)抽象方法怎诫,可以傳入lambda表達(dá)式。類似的贷痪,如果一個(gè)方法接受聲明于 java.util.function 包內(nèi)的接口幻妓,例如 Predicate、Function劫拢、Consumer 或 Supplier肉津,那么可以向其傳lambda表達(dá)式。

2)lambda表達(dá)式內(nèi)可以使用方法引用舱沧,僅當(dāng)該方法不修改lambda表達(dá)式提供的參數(shù)妹沙。本例中的lambda表達(dá)式可以換為方法引用,因?yàn)檫@僅是一個(gè)參數(shù)相同的簡(jiǎn)單方法調(diào)用狗唉。

list.forEach(n -> System.out.println(n));

list.forEach(System.out::println);? // 使用方法引用

然而初烘,若對(duì)參數(shù)有任何修改,則不能使用方法引用分俯,而需鍵入完整地lambda表達(dá)式肾筐,如下所示:

list.forEach((String s) -> System.out.println("*"+ s +"*"));

事實(shí)上,可以省略這里的lambda參數(shù)的類型聲明缸剪,編譯器可以從列表的類屬性推測(cè)出來(lái)吗铐。

3)lambda內(nèi)部可以使用靜態(tài)、非靜態(tài)和局部變量杏节,這稱為lambda內(nèi)的變量捕獲唬渗。

4)Lambda表達(dá)式在Java中又稱為閉包或匿名函數(shù),所以如果有同事把它叫閉包的時(shí)候奋渔,不用驚訝镊逝。

5)Lambda方法在編譯器內(nèi)部被翻譯成私有方法,并派發(fā)?invokedynamic?字節(jié)碼指令來(lái)進(jìn)行調(diào)用嫉鲸〕潘猓可以使用JDK中的?javap?工具來(lái)反編譯class文件。使用?javap -p?或?javap -c -v?命令來(lái)看一看lambda表達(dá)式生成的字節(jié)碼玄渗。大致應(yīng)該長(zhǎng)這樣:

private static java.lang.Object lambda$0(java.lang.String);

6)lambda表達(dá)式有個(gè)限制座菠,那就是只能引用?final?或?final?局部變量,這就是說(shuō)不能在lambda內(nèi)部修改定義在域外的變量藤树。

List primes = Arrays.asList(new Integer[]{2, 3,5,7});int factor = 2;primes.forEach(element -> { factor++; }); Compile time error :"local variables referenced from a lambda expression must be final or effectively final"

另外浴滴,只是訪問(wèn)它而不作修改是可以的,如下所示:

List primes = Arrays.asList(new Integer[]{2, 3,5,7});

int factor = 2;

primes.forEach(element -> { System.out.println(factor*element); });

輸出:

4

6

10

14

因此岁钓,它看起來(lái)更像不可變閉包升略,類似于Python微王。

以上就是Java 8的lambda表達(dá)式的全部10個(gè)例子。此次修改將成為Java史上最大的一次品嚣,將深遠(yuǎn)影響未來(lái)Java開發(fā)者使用集合框架的方式骂远。我想規(guī)模最相似的一次修改就是Java 5的發(fā)布了,它帶來(lái)了很多優(yōu)點(diǎn)腰根,提升了代碼質(zhì)量,例如:泛型拓型、枚舉额嘿、自動(dòng)裝箱(Autoboxing)、靜態(tài)導(dǎo)入劣挫、并發(fā)API和變量參數(shù)册养。上述特性使得Java代碼更加清晰,我想lambda表達(dá)式也將進(jìn)一步改進(jìn)它压固。我在期待著開發(fā)并行第三方庫(kù)球拦,這可以使高性能應(yīng)用變得更容易寫。

原文地址:https://juejin.im/post/5abc9ccc6fb9a028d6643eea

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末帐我,一起剝皮案震驚了整個(gè)濱河市坎炼,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌拦键,老刑警劉巖谣光,帶你破解...
    沈念sama閱讀 206,214評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異芬为,居然都是意外死亡萄金,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,307評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門媚朦,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)氧敢,“玉大人,你說(shuō)我怎么就攤上這事询张∷锕裕” “怎么了?”我有些...
    開封第一講書人閱讀 152,543評(píng)論 0 341
  • 文/不壞的土叔 我叫張陵瑞侮,是天一觀的道長(zhǎng)的圆。 經(jīng)常有香客問(wèn)我,道長(zhǎng)半火,這世上最難降的妖魔是什么越妈? 我笑而不...
    開封第一講書人閱讀 55,221評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮钮糖,結(jié)果婚禮上梅掠,老公的妹妹穿的比我還像新娘酌住。我一直安慰自己,他們只是感情好阎抒,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,224評(píng)論 5 371
  • 文/花漫 我一把揭開白布酪我。 她就那樣靜靜地躺著,像睡著了一般且叁。 火紅的嫁衣襯著肌膚如雪都哭。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,007評(píng)論 1 284
  • 那天逞带,我揣著相機(jī)與錄音欺矫,去河邊找鬼。 笑死展氓,一個(gè)胖子當(dāng)著我的面吹牛穆趴,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播遇汞,決...
    沈念sama閱讀 38,313評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼未妹,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了空入?” 一聲冷哼從身側(cè)響起络它,我...
    開封第一講書人閱讀 36,956評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎执庐,沒(méi)想到半個(gè)月后酪耕,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,441評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡轨淌,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,925評(píng)論 2 323
  • 正文 我和宋清朗相戀三年迂烁,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片递鹉。...
    茶點(diǎn)故事閱讀 38,018評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡盟步,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出躏结,到底是詐尸還是另有隱情却盘,我是刑警寧澤,帶...
    沈念sama閱讀 33,685評(píng)論 4 322
  • 正文 年R本政府宣布媳拴,位于F島的核電站黄橘,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏屈溉。R本人自食惡果不足惜塞关,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,234評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望子巾。 院中可真熱鬧帆赢,春花似錦小压、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,240評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至瘾婿,卻和暖如春蜻牢,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背偏陪。 一陣腳步聲響...
    開封第一講書人閱讀 31,464評(píng)論 1 261
  • 我被黑心中介騙來(lái)泰國(guó)打工孩饼, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人竹挡。 一個(gè)月前我還...
    沈念sama閱讀 45,467評(píng)論 2 352
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像立膛,于是被迫代替她去往敵國(guó)和親揪罕。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,762評(píng)論 2 345

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