JDK8的新特性——Lambda表達式

来源:https://www.cnblogs.com/yulinfeng/archive/2018/02/18/8452379.html
-Advertisement-
Play Games

JDK8已經發佈快4年的時間了,現在來談它的新特性顯得略微的有點“不合時宜”。儘管JDK8已不再“新”,但它的重要特性之一——Lambda表達式依然是不被大部分開發者所熟練運用,甚至不被開發者所熟知。 國內的開發環境大家都知道,有各種的老項目,有各種各樣的發佈風險,讓公司以及項目組對新的技術往往望而 ...


  JDK8已經發佈快4年的時間了,現在來談它的新特性顯得略微的有點“不合時宜”。儘管JDK8已不再“新”,但它的重要特性之一——Lambda表達式依然是不被大部分開發者所熟練運用,甚至不被開發者所熟知。

  國內的開發環境大家都知道,有各種的老項目,有各種各樣的發佈風險,讓公司以及項目組對新的技術往往望而卻步,有公司甚至時至今日還在使用JDK6來進行項目開發,這導致了在很多技術的選擇上受到了很大限制,進而不能跟隨時代的腳步使得項目甚至公司一步一步走向衰落。

  本文簡單認識JDK8的重要新特性之一——Lambda表達式。 在JDK8之前,Java是不支持函數式編程的,所謂的函數編程,即可理解是將一個函數(也稱為“行為”)作為一個參數進行傳遞。通常我們提及得更多的是面向對象編程,面向對象編程是對數據的抽象(各種各樣的POJO類),而函數式編程則是對行為的抽象(將行為作為一個參數進行傳遞)。在JavaScript中這是很常見的一個語法特性,但在Java中將一個函數作為參數傳遞這卻行不通,好在JDK8的出現打破了Java的這一限制。

 

認識Lambda表達式

  首先來引入一個示例,不知給是否有在IDEA編寫代碼的經歷,如果在JDK8的環境下如下所示按照Java傳統的語法規則編寫一個線程。

1 new Thread(new Runnable() {
2     @Override
3     public void run() {
4         System.out.println("Hello World!");
5     }
6 });

  IDEA會給出提示可以使用Lambda表達式替換。

  使用Lambda表達式則只需要使用一句話就可代替上面使用匿名類的方式。

new Thread(() -> System.out.println("Hello World!"));

   在這個例子中,傳統的語法規則,我們是將一個匿名內部類作為參數進行傳遞,我們實現了Runnable介面,並將其作為參數傳遞給Thread類,這實際上我們傳遞的是一段代碼,也即我們將代碼作為了數據進行傳遞,這就帶來許多不必要的“樣板代碼”。

  Lambda表達式一共有三部分組成:

  後面的示例中我們會詳解這個結構,包括有無參數,有無返回值的問題。 那麼這個看起來奇奇怪怪的不太像Java的語法規則,其本身含義到底什麼呢?這也是開始困擾我的問題,什麼時候在什麼場景下可以使用Lambda表達式。

  能夠接收Lambda表達式的參數類型,是一個只包含一個方法的介面。只包含一個方法的介面稱之為“函數介面”。

  例如上面創建一個線程的示例,Runnable介面只包含一個方法,所以它被稱為“函數介面”,所以它可以使用Lambad表達式來代替匿名內部類。根據這個規則,我們試著來寫一個函數介面,並使用Lambda表達式作為參數傳遞。

1 package com.coderbuff.custom;
2 
3 /**
4  * 函數介面:只有一個方法的介面。作為Lambda表達式的類型
5  * Created by Kevin on 2018/2/17.
6  */
7 public interface FunctionInterface {
8     void test();
9 }

   測試:

 1 package com.coderbuff.custom;
 2 
 3 import org.junit.Test;
 4 
 5 /**
 6  * 函數介面測試
 7  * Created by Kevin on 2018/2/17.
 8  */
 9 public class FunctionInterfaceTest {
10 
11     @Test
12     public void testLambda() {
13         func(new FunctionInterface() {
14             @Override
15             public void test() {
16                 System.out.println("Hello World!");
17             }
18         });
19         //使用Lambda表達式代替上面的匿名內部類
20         func(() -> System.out.println("Hello World"));
21     }
22 
23     private void func(FunctionInterface functionInterface) {
24         functionInterface.test();
25     }
26 }

  可以看到,只要是一個介面中只包含一個方法,則可以使用Lambda表達式,這樣的介面稱之為“函數介面”。

  上面的函數介面比較簡單不包含參數,也不包含返回值

  我們再來修改FunctionInterface函數介面逐步加大Lambda表達式的難度——包含參數,不包含返回值

1 package com.coderbuff.custom;
2 
3 /**
4  * 函數介面:只有一個方法的介面。作為Lambda表達式的類型
5  * Created by Kevin on 2018/2/17.
6  */
7 public interface FunctionInterface {
8     void test(int param);
9 }

  測試:

 1 package com.coderbuff.custom;
 2 
 3 import org.junit.Test;
 4 
 5 /**
 6  * 函數介面測試
 7  * Created by Kevin on 2018/2/17.
 8  */
 9 public class FunctionInterfaceTest {
10 
11     @Test
12     public void testLambda() {
13         //使用Lambda表達式代替匿名內部類
14         func((x) -> System.out.println("Hello World" + x));
15     }
16 
17     private void func(FunctionInterface functionInterface) {
18         int x = 1;
19         functionInterface.test(x);
20     }
21 }

  關註Lambda表達式“(x) -> Sysout.out.println("Hello World" + x)”,左邊傳遞的是參數,此處並沒有指明參數類型,因為它可以通過上下文進行類型推導,但在有些情況下不能推導出參數類型(在編譯時不能推導通常IDE會提示),此時則需要指明參數類型。我個人建議,任何情況下指明函數的參數類型

  哪種情況不能推導出參數類型呢?就是函數介面是一個泛型的時候。

1 package com.coderbuff.custom;
2 
3 /**
4  * 函數介面:只有一個方法的介面。作為Lambda表達式的類型
5  * Created by Kevin on 2018/2/17.
6  */
7 public interface FunctionInterface<T> {
8     void test(T param);
9 } 

  測試:

 1 package com.coderbuff.custom;
 2 
 3 import org.junit.Test;
 4 
 5 /**
 6  * 函數介面測試
 7  * Created by Kevin on 2018/2/17.
 8  */
 9 public class FunctionInterfaceTest {
10 
11     @Test
12     public void testLambda() {
13         //使用Lambda表達式代替匿名內部類
14         func((Integer x) -> System.out.println("Hello World" + x));
15     }
16 
17     private void func(FunctionInterface<Integer> functionInterface) {
18         int x = 1;
19         functionInterface.test(x);
20     }
21 }

  上面的示例提到了Lambda表達式的兩種情況:

  無參數,無返回值;

  有參數,無返回值。

  接下來就是有參數,有返回值這種較為複雜的情況。

1 package com.coderbuff.custom;
2 
3 /**
4  * 函數介面:只有一個方法的介面。作為Lambda表達式的類型
5  * Created by Kevin on 2018/2/17.
6  */
7 public interface FunctionInterface<T> {
8     boolean test(T param);
9 }

   測試:

 1 package com.coderbuff.custom;
 2 
 3 import org.junit.Test;
 4 
 5 /**
 6  * 函數介面測試
 7  * Created by Kevin on 2018/2/17.
 8  */
 9 public class FunctionInterfaceTest {
10 
11     @Test
12     public void testLambda() {
13         //使用Lambda表達式代替匿名內部類
14         func((Integer x) -> true);
15     }
16 
17     private void func(FunctionInterface<Integer> functionInterface) {
18         int x = 1;
19         functionInterface.test(x);
20     }
21 }

  此時的Lambda表達式“(Integer x) -> true”,右邊是表達式的主體,直接返回true,如果有多行代碼,則可以直接使用花括弧表示,例如:

func((Integer x) -> {
    System.out.println("Hello World" + x);
    return true;
});

  Lambda表達式基本的語法規則:

  無參數,無返回值;

  有參數,無返回值;

  有參數,有返回值。

  這三種基本情況已經大致清楚了,特別是需要弄清,什麼時候可以使用Lambda表達式代替匿名內部類,也就是Lambda表達式的應用場景是函數介面。Lambda表達式這一新特性在JDK8中的引入,更大的好處則是集合API的更新,新增的Stream類庫,使得我們在遍歷使用集合時不再像以往那樣不斷地使用for迴圈。

 

JDK8使用集合的正確姿勢

  示例:計算來自“chengdu”的學生數量有多少。

  在JDK8前的代碼:

for (Student student : studentList) {
    if (student.getCity().equals("chengdu")) {
        count++;
    }
}

  JDK8使用集合的正確姿勢:

count = studentList.stream().filter((student -> student.getCity().equals("chengdu"))).count();

  API的使用“難度”恰似提高了,實際只是不熟悉而已。傳統迭代的方式需要閱讀完整個迴圈才能明白代碼邏輯,JDK8通過流的方式則可以望文生義且代碼量大大減小。

  其中最為重要的是——Stream流。Stream的是通過函數式編程方式實現的在集合類上進行複雜操作的工具。若要詳細講解Stream的實現方式我相信再寫一篇博客也不為過,所以此處不再考查Stream的內部實現。這裡是想告訴大家,如果有幸使用JDK8的開發環境進行開發,儘量學習使用新的集合操作API。

  上面對於Lambda表達式以及函數式編程僅僅只是到了一個“認識”的地步,似乎只是感受到了縮小代碼量,本文對於Lambda式的認識不深入更多的是對於後面更多的知識做一個鋪墊或者作為一個掃盲貼,有關Lambda表達式的應用太多,併發編程、響應式編程等等。如果你有關於Lambda表達式或者函數式編程有更好的見解不妨留下評論。

 

 

這是一個能給程式員加buff的公眾號 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • NFS(Network File System)即網路文件系統,是FreeBSD支持的文件系統中的一種,它允許網路中的電腦之間通過TCP/IP網路資源共用。在NFS的應用中,本地NFS的客戶端應用可以透明地讀寫位於遠端NFS伺服器上的文件,就像訪問本地文件一樣。 1、好處 (1)節省本地存儲空間, ...
  • 命令模式下輸入如下命令可實現替換: ~~~~ s/str1/str2/ 替換當前行第一個 str1 為 str2 s/str1/str2/g 替換當前行中所有的 str1 為 str2 m,ns/str1/str2/ 替換第 m 行到第 n 行中每一行的第一個 str1 為 str2 m,ns/st ...
  • 文檔版本號:20180216最近在Ubuntu Linux 14.04上和CentOS Linux 7.4上成功安裝了Harbor,現將過程整理如下,供大家參考: 備註:使用非root用戶操作Docker,需要創建docker組 sudo groupadd docker 將當前用戶加入docker組 ...
  • wget命令用來從指定的URL下載文件。wget非常穩定,它在帶寬很窄的情況下和不穩定網路中有很強的適應性,如果是由於網路的原因下載失敗,wget會不斷的嘗試,直到整個文件下載完畢。如果是伺服器打斷下載過程,它會再次聯到伺服器上從停止的地方繼續下載。這對從那些限定了鏈接時間的伺服器上下載大文件非常有 ...
  • 本文目錄:1. LVS簡介2. LVS-ipvs三種模式的工作原理 2.1 VS/NAT模式 2.2 VS/TUN模式 2.3 VS/DR模式 2.4 lvs-ipvs的三種模式比較3. VS/TUN和VS/DR模式中的ARP問題4. LVS負載均衡的調度演算法 網站架構中,負載均衡技術是實現網站架構 ...
  • PAM(Pluggable Authentication Modules) PAM架構 pam 認證原理 PAM 認證過程 PAM的配置文件 pam文檔 限制的實現方式 共有3種方式: 1)通過ulimit命令 2)在/etc/security/limits.d/目錄下創建限制文件來實現 3)修改/ ...
  • AIDE ...
  • JavaScript的數據類型 基本區分方法 ECMAScript標准定義了7種數據類型 6 種 基本類型: Boolean,兩種取值:true和false Null,一種取值:null Undefined,一種取值:undefined Number,JS的數值為基於 IEEE 754 標準的雙精度 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...