微服務為什么要容器化?
Lambda 表達式是 Java 8 引入的一種簡潔的表示匿名方法的方式,使用它可以用于替代某些匿名內部類對象,從而讓程序更簡潔,可讀性更好。但 Lambda 表達式的底層是如何實現的呢?接下來我們一起來看。
1.未Lambda表達式
未使用 Lambda 表達式之前,我們創建一個線程,可以這樣寫:
Thread t1 = new Thread(new Runnable() {
@Override
public void run() {
System.out.println("t1");
}
});
t1.start();
其中 Runnable 匿名內部類,查看 Runnable 源碼,我們可以看到 Runnable 的實現如下:
圖片
1.1 什么是匿名內部類?
匿名內部類是在 Java 中定義的一個沒有名稱的內部類。它通常在一個類的成員位置或者方法體內直接定義,并且立即實例化。
匿名內部類的主要用途在于簡化代碼,避免為了實現一個簡單的功能而定義一個完整的類。它特別適用于只需要一次使用的類,比如實現一個接口的單方法(即函數式接口)的場合。
PS:自從 Java 8 引入 Lambda 表達式后,很多原本使用匿名內部類的地方可以被更簡潔的 Lambda 表達式替代。
上面代碼中的 new Runnable 就是一個標準匿名內部類的使用。
1.2 什么是@FunctionalInterface?
@FunctionalInterface 是 Java 8 引入的一個注解,它用于標記一個接口為函數式接口。
函數式接口是指只包含一個抽象方法的接口。這個注解雖然不是必需的,但它提供了一種明確的方式告訴編譯器和開發者,這個接口是設計為函數式接口的。
@FunctionalInterface 注解的作用如下:
- 編譯時檢查:當一個接口被標記為 @FunctionalInterface 時,編譯器會檢查該接口是否只有一個抽象方法。如果不符合函數式接口的定義(即存在多個抽象方法),編譯器會報錯,提醒開發者修正。這為開發者提供了明確的編譯時保障,確保所標記的接口確實符合函數式接口的要求。
- 代碼明確性:即使不加 @FunctionalInterface 注解,只要接口符合函數式接口的定義,它仍然可以被視為函數式接口。但注解的存在增加了代碼的明確性和可讀性,使得其他開發者更容易理解該接口的設計意圖。
- 支持 Lambda 表達式:函數式接口的主要目的是為了支持 Lambda 表達式。通過 Lambda 表達式,開發者可以以更簡潔的方式實現函數式接口的抽象方法,從而減少模板代碼,使代碼更加簡潔和易于理解。由于 Lambda 表達式本身不包含類型信息,Java 編譯器需要一種機制來確定 Lambda 表達式對應的目標類型。函數式接口就扮演了這一角色——Lambda 表達式可以被賦值給任何兼容的函數式接口類型,編譯器會依據接口的唯一抽象方法來推斷 Lambda 表達式的參數類型和返回類型。
在 Java 標準庫中,有許多使用 @FunctionalInterface 注解的接口,如 java.util.function 包下的 Function、Predicate、Consumer 等,這些接口都是函數式接口,廣泛用于數據處理、過濾、轉換等操作。此外,在 Spring Boot 框架中,也經常使用函數式接口來定義事件監聽器、回調函數等。
2.使用Lambda表達式
未使用 Lambda 表達式之前,我們創建一個線程是這樣寫的:
Thread t1 = new Thread(new Runnable() {
@Override
public void run() {
System.out.println("t1");
}
});
t1.start();
而用了 Lambda 表達式,我們可以這樣寫:
Thread t1 = new Thread(() -> { System.out.println("t1"); });
t1.start();
從上述代碼可以看出,當我們使用 Lambda 表達式之后,代碼就變得更簡潔和優雅了。
3.Lambda詳解
Lambda 表達式的語法形式如下:
(parameters) -> expression
或者是:
(parameters) -> { statements; }
以上語法含義如下:
- 參數列表:在圓括號內的部分,用于定義傳遞給 Lambda 體的參數。參數列表可以為空,也可以包含多個參數,參數之間用逗號隔開。
- 箭頭符號:是 Lambda 表達式的分隔符,將參數列表與表達式或語句塊分隔開。
- Lambda 體:包含了具體的執行邏輯,可以是一個表達式或是一個由多個語句組成的代碼塊。
3.1 使用場景
Lambda 表達式主要用于執行函數式接口(Function Interface),即只有一個抽象方法的接口。常見的函數式接口包括 java.util.function 包下的 Predicate、Function、Consumer 等。
3.2 舉個例子
假設我們有一個List,并且我們想要對這個列表進行過濾操作,只保留偶數元素。使用 Lambda 表達式可以非常方便地實現這一功能:
import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;
public class Main {
public static void main(String[] args) {
List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5, 6, 7, 8, 9);
// 使用 Lambda 表達式過濾出偶數
List<Integer> evenNumbers = numbers.stream()
.filter(n -> n % 2 == 0)
.collect(Collectors.toList());
System.out.println(evenNumbers); // 輸出 [2, 4, 6, 8]
}
}
在這個例子中,n -> n % 2 == 0 是一個 Lambda 表達式,它接受一個整數 n 作為輸入參數,并返回一個布爾值。這個 Lambda 表達式被用作 filter 方法的參數,該方法期望一個 Predicate類型的函數式接口實例。
4.Lambda底層原理
Lambda 底層運行原理如下:
在程序運行時,會在類中生成一個匿名內部類,匿名內部類會實現接口,并重寫接口中的抽象方法。
類中會生成一個靜態方法,靜態方法中的代碼就是 Lambda 表達式中的代碼。
匿名內部類重寫的抽象方法,會調用上一步的靜態方法,從而實現 Lambda 代碼的執行。
所以,綜合來說,Lambda 表達式其實是匿名內部類的語法糖,這個語法糖在程序執行時會進行兌現,也就是生成匿名內部類并進行任務執行。