Lambda 表達式有何用處?史上最好的java 函數編程介紹 強烈推薦!!!

 
 
做者:Sevenvidia
連接:https://www.zhihu.com/question/20125256/answer/324121308
來源:知乎
著做權歸做者全部。商業轉載請聯繫做者得到受權,非商業轉載請註明出處。

答主對Java比較熟悉,就用Java來說一講吧。html

<img src="https://pic3.zhimg.com/50/v2-c7cb18cbd37262ffad7b04e50fb13b92_hd.jpg" data-caption="" data-size="normal" data-rawwidth="600" data-rawheight="61" class="origin_image zh-lightbox-thumb" width="600" data-original="https://pic3.zhimg.com/v2-c7cb18cbd37262ffad7b04e50fb13b92_r.jpg">

什麼是Lambda?

咱們知道,對於一個Java變量,咱們能夠賦給其一個「值」java

<img src="https://pic3.zhimg.com/50/v2-ab6545c49383236a4af3f28a47886090_hd.jpg" data-caption="" data-size="normal" data-rawwidth="602" data-rawheight="204" class="origin_image zh-lightbox-thumb" width="602" data-original="https://pic3.zhimg.com/v2-ab6545c49383236a4af3f28a47886090_r.jpg">

若是你想把「一塊代碼」賦給一個Java變量,應該怎麼作呢?express

好比,我想把右邊那塊代碼,賦給一個叫作aBlockOfCode的Java變量:api

<img src="https://pic3.zhimg.com/50/v2-1cc87e82fba0872c2cae3fee08e8fe41_hd.jpg" data-caption="" data-size="normal" data-rawwidth="534" data-rawheight="133" class="origin_image zh-lightbox-thumb" width="534" data-original="https://pic3.zhimg.com/v2-1cc87e82fba0872c2cae3fee08e8fe41_r.jpg">

在Java 8以前,這個是作不到的。可是Java 8問世以後,利用Lambda特性,就能夠作到了。oracle

<img src="https://pic3.zhimg.com/50/v2-145a556d86806c3163391a13428e3f03_hd.jpg" data-caption="" data-size="normal" data-rawwidth="585" data-rawheight="120" class="origin_image zh-lightbox-thumb" width="585" data-original="https://pic3.zhimg.com/v2-145a556d86806c3163391a13428e3f03_r.jpg">

固然,這個並非一個很簡潔的寫法。因此,爲了使這個賦值操做更加elegant, 咱們能夠移除一些沒用的聲明。app

<img src="https://pic1.zhimg.com/50/v2-a712753b42972e094a548ae02fa82987_hd.jpg" data-caption="" data-size="normal" data-rawwidth="1362" data-rawheight="1498" class="origin_image zh-lightbox-thumb" width="1362" data-original="https://pic1.zhimg.com/v2-a712753b42972e094a548ae02fa82987_r.jpg">

這樣,咱們就成功的很是優雅的把「一塊代碼」賦給了一個變量。而「這塊代碼」,或者說「這個被賦給一個變量的函數」,就是一個Lambda表達式函數

可是這裏仍然有一個問題,就是變量aBlockOfCode的類型應該是什麼?oop

在Java 8裏面,全部的Lambda的類型都是一個接口,而Lambda表達式自己,也就是」那段代碼「,須要是這個接口的實現。這是我認爲理解Lambda的一個關鍵所在,簡而言之就是,Lambda表達式自己就是一個接口的實現。直接這樣說可能仍是有點讓人困擾,咱們繼續看看例子。咱們給上面的aBlockOfCode加上一個類型:學習

<img src="https://pic1.zhimg.com/50/v2-55de66060b4cb70193ddc7fea201b257_hd.jpg" data-caption="" data-size="normal" data-rawwidth="534" data-rawheight="244" class="origin_image zh-lightbox-thumb" width="534" data-original="https://pic1.zhimg.com/v2-55de66060b4cb70193ddc7fea201b257_r.jpg">

這種只有一個接口函數須要被實現的接口類型,咱們叫它」函數式接口「。爲了不後來的人在這個接口中增長接口函數致使其有多個接口函數須要被實現,變成"非函數接口」,咱們能夠在這個上面加上一個聲明@FunctionalInterface, 這樣別人就沒法在裏面添加新的接口函數了:ui

<img src="https://pic1.zhimg.com/50/v2-2c57e7411de227d1eb09c327d01fb766_hd.jpg" data-caption="" data-size="normal" data-rawwidth="318" data-rawheight="123" class="content_image" width="318">

這樣,咱們就獲得了一個完整的Lambda表達式聲明:

<img src="https://pic4.zhimg.com/50/v2-02eedc528fcee115f5ed0b7b045846d7_hd.jpg" data-caption="" data-size="normal" data-rawwidth="1354" data-rawheight="148" class="origin_image zh-lightbox-thumb" width="1354" data-original="https://pic4.zhimg.com/v2-02eedc528fcee115f5ed0b7b045846d7_r.jpg"> <img src="https://pic4.zhimg.com/50/v2-b3c9ad03a5333f5e61c3ee8186210634_hd.jpg" data-caption="" data-size="normal" data-rawwidth="600" data-rawheight="61" class="origin_image zh-lightbox-thumb" width="600" data-original="https://pic4.zhimg.com/v2-b3c9ad03a5333f5e61c3ee8186210634_r.jpg">

Lambda表達式有什麼做用?

最直觀的做用就是使得代碼變得異常簡潔。

咱們能夠對比一下Lambda表達式和傳統的Java對同一個接口的實現:

<img src="https://pic1.zhimg.com/50/v2-dbd46cf9d188d0fde25db700c23dcc79_hd.jpg" data-caption="" data-size="normal" data-rawwidth="1178" data-rawheight="403" class="origin_image zh-lightbox-thumb" width="1178" data-original="https://pic1.zhimg.com/v2-dbd46cf9d188d0fde25db700c23dcc79_r.jpg">

這兩種寫法本質上是等價的。可是顯然,Java 8中的寫法更加優雅簡潔。而且,因爲Lambda能夠直接賦值給一個變量,咱們就能夠直接把Lambda做爲參數傳給函數, 而傳統的Java必須有明確的接口實現的定義,初始化才行:

<img src="https://pic1.zhimg.com/50/v2-28606f4328308baf7f70a36bd689e5ea_hd.jpg" data-caption="" data-size="normal" data-rawwidth="1112" data-rawheight="498" class="origin_image zh-lightbox-thumb" width="1112" data-original="https://pic1.zhimg.com/v2-28606f4328308baf7f70a36bd689e5ea_r.jpg">

有些狀況下,這個接口實現只須要用到一次。傳統的Java 7必需要求你定義一個「污染環境」的接口實現MyInterfaceImpl,而相較之下Java 8的Lambda, 就顯得乾淨不少。

<img src="https://pic1.zhimg.com/50/v2-413d712fb74b8406502af05031cf8d4c_hd.jpg" data-caption="" data-size="normal" data-rawwidth="600" data-rawheight="61" class="origin_image zh-lightbox-thumb" width="600" data-original="https://pic1.zhimg.com/v2-413d712fb74b8406502af05031cf8d4c_r.jpg">

Lambda結合FunctionalInterface Lib, forEach, stream(),method reference等新特性可使代碼變的更加簡潔!

直接上例子。

假設Person的定義和List<Person>的值都給定。

&amp;lt;img src="https://pic3.zhimg.com/50/v2-02ae14b4c341e2303ae86e2da6a038a2_hd.jpg" data-caption="" data-size="normal" data-rawwidth="693" data-rawheight="236" class="origin_image zh-lightbox-thumb" width="693" data-original="https://pic3.zhimg.com/v2-02ae14b4c341e2303ae86e2da6a038a2_r.jpg"&amp;gt;

如今須要你打印出guiltyPersons List裏面全部LastName以"Z"開頭的人的FirstName。

原生態Lambda寫法:定義兩個函數式接口,定義一個靜態函數,調用靜態函數並給參數賦值Lambda表達式。

&amp;lt;img src="https://pic4.zhimg.com/50/v2-fdef41934be8804fa244e89c84a567f1_hd.jpg" data-caption="" data-size="normal" data-rawwidth="587" data-rawheight="457" class="origin_image zh-lightbox-thumb" width="587" data-original="https://pic4.zhimg.com/v2-fdef41934be8804fa244e89c84a567f1_r.jpg"&amp;gt;

這個代碼實際上已經比較簡潔了,可是咱們還能夠更簡潔麼?

固然能夠。在Java 8中有一個函數式接口的包,裏面定義了大量可能用到的函數式接口(java.util.function (Java Platform SE 8 ))。因此,咱們在這裏壓根都不須要定義NameChecker和Executor這兩個函數式接口,直接用Java 8函數式接口包裏的Predicate<T>和Consumer<T>就能夠了——由於他們這一對的接口定義和NameChecker/Executor實際上是同樣的。

&amp;lt;img src="https://pic2.zhimg.com/50/v2-ad5b3e8d225bf9e6d988c6dc83819637_hd.jpg" data-caption="" data-size="normal" data-rawwidth="551" data-rawheight="247" class="origin_image zh-lightbox-thumb" width="551" data-original="https://pic2.zhimg.com/v2-ad5b3e8d225bf9e6d988c6dc83819637_r.jpg"&amp;gt;

第一步簡化 - 利用函數式接口包:

&amp;lt;img src="https://pic4.zhimg.com/50/v2-493168d1a72120d69b0d13b4711c3c9f_hd.jpg" data-caption="" data-size="normal" data-rawwidth="609" data-rawheight="323" class="origin_image zh-lightbox-thumb" width="609" data-original="https://pic4.zhimg.com/v2-493168d1a72120d69b0d13b4711c3c9f_r.jpg"&amp;gt;

靜態函數裏面的for each循環實際上是很是礙眼的。這裏能夠利用Iterable自帶的forEach()來替代。forEach()自己能夠接受一個Consumer<T> 參數。

第二步簡化 - 用Iterable.forEach()取代foreach loop:

&amp;lt;img src="https://pic1.zhimg.com/50/v2-bdd6f5a6dbf65b578080f7b7ad5ab6cf_hd.jpg" data-caption="" data-size="normal" data-rawwidth="606" data-rawheight="276" class="origin_image zh-lightbox-thumb" width="606" data-original="https://pic1.zhimg.com/v2-bdd6f5a6dbf65b578080f7b7ad5ab6cf_r.jpg"&amp;gt;

因爲靜態函數其實只是對List進行了一通操做,這裏咱們能夠甩掉靜態函數,直接使用stream()特性來完成。stream()的幾個方法都是接受Predicate<T>,Consumer<T>等參數的(java.util.stream (Java Platform SE 8 ))。你理解了上面的內容,stream()這裏就很是好理解了,並不須要多作解釋。

第三步簡化 - 利用stream()替代靜態函數:

&amp;lt;img src="https://pic1.zhimg.com/50/v2-e196d987f852b9b8e26a6a9dac648a06_hd.jpg" data-caption="" data-size="normal" data-rawwidth="507" data-rawheight="116" class="origin_image zh-lightbox-thumb" width="507" data-original="https://pic1.zhimg.com/v2-e196d987f852b9b8e26a6a9dac648a06_r.jpg"&amp;gt;

對比最開始的Lambda寫法,這裏已經很是很是簡潔了。可是若是,咱們的要求變一下,變成print這我的的所有信息,及p -> System.out.println(p); 那麼還能夠利用Method reference來繼續簡化。所謂Method reference, 就是用已經寫好的別的Object/Class的method來代替Lambda expression。格式以下:

&amp;lt;img src="https://pic3.zhimg.com/50/v2-12622326a5682285ce235d96291f3bb8_hd.jpg" data-caption="" data-size="normal" data-rawwidth="459" data-rawheight="76" class="origin_image zh-lightbox-thumb" width="459" data-original="https://pic3.zhimg.com/v2-12622326a5682285ce235d96291f3bb8_r.jpg"&amp;gt;

第四步簡化 - 若是是println(p),則能夠利用Method reference代替forEach中的Lambda表達式:

&amp;lt;img src="https://pic4.zhimg.com/50/v2-f29e6569d0265b91794565ae81d54265_hd.jpg" data-caption="" data-size="normal" data-rawwidth="449" data-rawheight="104" class="origin_image zh-lightbox-thumb" width="449" data-original="https://pic4.zhimg.com/v2-f29e6569d0265b91794565ae81d54265_r.jpg"&amp;gt;

這基本上就是能寫的最簡潔的版本了。

&amp;lt;img src="https://pic1.zhimg.com/50/v2-ad2b43af522ea121d3b751c4a0f881c6_hd.jpg" data-caption="" data-size="normal" data-rawwidth="600" data-rawheight="61" class="origin_image zh-lightbox-thumb" width="600" data-original="https://pic1.zhimg.com/v2-ad2b43af522ea121d3b751c4a0f881c6_r.jpg"&amp;gt;

Lambda配合Optional<T>可使Java對於null的處理變的異常優雅

這裏假設咱們有一個person object,以及一個person object的Optional wrapper:

&amp;lt;img src="https://pic3.zhimg.com/50/v2-e78d71b9699504e46d86d77aabec42cf_hd.jpg" data-caption="" data-size="normal" data-rawwidth="539" data-rawheight="134" class="origin_image zh-lightbox-thumb" width="539" data-original="https://pic3.zhimg.com/v2-e78d71b9699504e46d86d77aabec42cf_r.jpg"&amp;gt;

Optional<T>若是不結合Lambda使用的話,並不能使原來繁瑣的null check變的簡單。

&amp;lt;img src="https://pic1.zhimg.com/50/v2-ce82e2c3e69caab9acdec9d4e42cde0e_hd.jpg" data-caption="" data-size="normal" data-rawwidth="667" data-rawheight="181" class="origin_image zh-lightbox-thumb" width="667" data-original="https://pic1.zhimg.com/v2-ce82e2c3e69caab9acdec9d4e42cde0e_r.jpg"&amp;gt;

只有當Optional<T>結合Lambda一塊兒使用的時候,才能發揮出其真正的威力!

咱們如今就來對比一下下面四種常見的null處理中,Java 8的Lambda+Optional<T>和傳統Java二者之間對於null的處理差別。

狀況一 - 存在則開幹

&amp;lt;img src="https://pic1.zhimg.com/50/v2-f41305009c93effa8fe047631d5342ed_hd.jpg" data-caption="" data-size="normal" data-rawwidth="747" data-rawheight="160" class="origin_image zh-lightbox-thumb" width="747" data-original="https://pic1.zhimg.com/v2-f41305009c93effa8fe047631d5342ed_r.jpg"&amp;gt;

狀況二 - 存在則返回,無則返回屁

&amp;lt;img src="https://pic4.zhimg.com/50/v2-cfa0e3298ba94efa29dc9cb8b32356fe_hd.jpg" data-caption="" data-size="normal" data-rawwidth="790" data-rawheight="181" class="origin_image zh-lightbox-thumb" width="790" data-original="https://pic4.zhimg.com/v2-cfa0e3298ba94efa29dc9cb8b32356fe_r.jpg"&amp;gt;

狀況三 - 存在則返回,無則由函數產生

&amp;lt;img src="https://pic2.zhimg.com/50/v2-4091bb966ac575fd83d5fa07dd7c2dce_hd.jpg" data-caption="" data-size="normal" data-rawwidth="996" data-rawheight="176" class="origin_image zh-lightbox-thumb" width="996" data-original="https://pic2.zhimg.com/v2-4091bb966ac575fd83d5fa07dd7c2dce_r.jpg"&amp;gt;

狀況四 - 奪命連環null檢查

&amp;lt;img src="https://pic1.zhimg.com/50/v2-7bc0ac86cd29a1cea92a853d73f41cec_hd.jpg" data-caption="" data-size="normal" data-rawwidth="867" data-rawheight="276" class="origin_image zh-lightbox-thumb" width="867" data-original="https://pic1.zhimg.com/v2-7bc0ac86cd29a1cea92a853d73f41cec_r.jpg"&amp;gt;

由上述四種狀況能夠清楚地看到,Optional<T>+Lambda可讓咱們少寫不少ifElse塊。尤爲是對於狀況四那種奪命連環null檢查,傳統java的寫法顯得冗長難懂,而新的Optional<T>+Lambda則清新脫俗,清楚簡潔。

&amp;lt;img src="https://pic4.zhimg.com/50/v2-e6b653cd6eced03376d5feb2e5943be3_hd.jpg" data-caption="" data-size="normal" data-rawwidth="600" data-rawheight="61" class="origin_image zh-lightbox-thumb" width="600" data-original="https://pic4.zhimg.com/v2-e6b653cd6eced03376d5feb2e5943be3_r.jpg"&amp;gt;

關於Java的Lambda, 還有東西須要討論和學習。好比如何handle lambda exception,如何利用Lambda的特性來進行parallel processing等。總之,我只是一如既往地介紹個大概,讓你大概知道,哦!原來是這樣子就OK了。網上關於Lambda有不少相關的教程,多看多練。假以時日,一定有所精益。

相關文章
相關標籤/搜索