最近在編寫一些Java業(yè)務(wù),使用了 Lambda語法,所以有必要整理一下lambda相關(guān)的知識,以便能夠加深理解。
- 什么是lambda表達式?
- lambda語法格式
- lambda 表達式重要特征
- 函數(shù)式接口
- lambda使用
- 匿名類和lambda中this指針
- 參考網(wǎng)站
1、什么是lambda表達式?
Lambda 表達式,也可稱為閉包,它是推動 Java 8 發(fā)布的最重要新特性。
Lambda 允許把函數(shù)作為一個方法的參數(shù)(函數(shù)作為參數(shù)傳遞進方法中)。
使用 Lambda 表達式可以使代碼變的更加簡潔緊湊。
2、lambda語法格式
(parameters) -> expression
(parameters) -> {expression;}
3、lambda 表達式重要特征
- 可選類型聲明:不需要聲明參數(shù)類型,編譯器可以統(tǒng)一識別參數(shù)值。
- 可選的參數(shù)圓括號:一個參數(shù)無需定義圓括號,但多個參數(shù)需要定義圓括號。
- 可選的大括號:如果主體包含了一個語句,就不需要使用大括號。
- 可選的返回關(guān)鍵字:如果主體只有一個表達式返回值則編譯器會自動返回值,大括號需要指定明表達式返回了一個數(shù)值。
4、函數(shù)式接口
-
什么是函數(shù)式接口:
- 函數(shù)式接口(Functional Interface)是Java 8對一類特殊類型的接口的稱呼。 這類接口只定義了唯一的抽象方法的接口(除了隱含的Object對象的公共方法), 因此最開始也就做SAM類型的接口(Single Abstract Method)。
- 函數(shù)式接口中可以額外定義Object中多個抽象方法,但這些抽象方法簽名必須和Object的 public 方法一樣
-
Java8新增函數(shù)式接口
- Predicate -- 傳入一個參數(shù),返回一個bool結(jié)果, 方法為boolean test(T t)
- Consumer -- 傳入一個參數(shù),無返回值,純消費。 方法為void accept(T t)
- Function -- 傳入一個參數(shù),返回一個結(jié)果,方法為R apply(T t)
- Supplier -- 無參數(shù)傳入,返回一個結(jié)果,方法為T get()
- UnaryOperator -- 一元操作符,繼承Function,傳入?yún)?shù)的類型和返回類型相同。
- BinaryOperator -- 二元操作符, 傳入的兩個參數(shù)的類型和返回類型相同, 繼承BiFunction
5、lambda使用
- 循環(huán)打印List<Integer>中的值
- 老方法:for(Integer i: list) { System.out.println(i);}
- Lambda表達式:list.forEach(x->System.out.println(x));
public static void main(String... args) {
List<Integer> list = Arrays.asList(10, 5, 4, 1, 76);
for(Integer i: list) {
System.out.println(i);
}
list.forEach(x -> System.out.println(x));
}
-反編譯代碼
for:
45: aload_1
46: invokeinterface #5, 1 // InterfaceMethod java/util/List.iterator:()Ljava/util/I
terator;
51: astore_2
52: aload_2
53: invokeinterface #6, 1 // InterfaceMethod java/util/Iterator.hasNext:()Z
58: ifeq 81
61: aload_2
62: invokeinterface #7, 1 // InterfaceMethod java/util/Iterator.next:()Ljava/lang/O
bject;
67: checkcast #2 // class java/lang/Integer
70: astore_3
71: getstatic #8 // Field java/lang/System.out:Ljava/io/PrintStream;
74: aload_3
75: invokevirtual #9 // Method java/io/PrintStream.println:(Ljava/lang/Object;
)V
78: goto 52
lambda:
81: aload_1
82: invokedynamic #10, 0 // InvokeDynamic #0:accept:()Ljava/util/function/Consumer
;
87: invokeinterface #11, 2 // InterfaceMethod java/util/List.forEach:(Ljava/util/fun
ction/Consumer;)V
private static void lambda$main$0(java.lang.Integer);
Code:
0: getstatic #8 // Field java/lang/System.out:Ljava/io/PrintStream;
3: aload_0
4: invokevirtual #9 // Method java/io/PrintStream.println:(Ljava/lang/Object;
)V
7: return
- lambda 編譯的時候生成了一個動態(tài)方法,調(diào)用也是執(zhí)行命令invokedynamic
6、匿名類和lambda中this指針
- 示例代碼
public void test1(){
new Thread(new Runnable() {
@Override
public void run() {
System.out.println(this.toString());
}
}).start();
new Thread(()->System.out.println(this.toString())).start();
}
- 反編譯
public class com.yuxiu.test.Main {
public void test1();
Code:
0: new #5 // class java/lang/Thread
3: dup
4: new #6 // class com/yuxiu/test/Main$1
7: dup
8: aload_0
9: invokespecial #7 // Method com/yuxiu/test/Main$1."<init>":(Lcom/yuxiu/test
/Main;)V
12: invokespecial #8 // Method java/lang/Thread."<init>":(Ljava/lang/Runnable;
)V
15: invokevirtual #9 // Method java/lang/Thread.start:()V
18: new #5 // class java/lang/Thread
21: dup
22: aload_0
23: invokedynamic #10, 0 // InvokeDynamic #0:run:(Lcom/yuxiu/test/Main;)Ljava/lang
/Runnable;
28: invokespecial #8 // Method java/lang/Thread."<init>":(Ljava/lang/Runnable;
)V
31: invokevirtual #9 // Method java/lang/Thread.start:()V
34: return
private void lambda$test1$0();
Code:
0: getstatic #11 // Field java/lang/System.out:Ljava/io/PrintStream;
3: aload_0
4: invokevirtual #12 // Method java/lang/Object.toString:()Ljava/lang/String;
7: invokevirtual #13 // Method java/io/PrintStream.println:(Ljava/lang/String;
)V
10: return
}
- 結(jié)論:
匿名類this指向匿名類
lambda中this指向lambda外的類