Java的Lambda表示式
1. 什麼是Lambda表示式
簡單的說,Lambda表示式就是匿名方法。Lambda表示式讓程式設計師能夠使用更加簡潔的程式碼,但是同樣也使程式碼的可讀性比較差。
Lambda表示式也叫做匿名方法或者閉包。
2. 和匿名內部類做對比
Lambda是匿名方法,這個時候我們會想想到匿名內部類,我們來回想一下匿名內部類的用法,比如下面的程式碼就是使用匿名內部類實現了一個執行緒。
public class Test {
public static void main(String[] args) {
Thread t = new Thread(new Runnable() {
@Override
public void run() {
System.out.println("執行緒:" + Thread.currentThread().getName());
}
});
t.start();
}
}
我們一般的做法是寫一個Runnable介面的實現類,然後new一個實現類再傳給Thread的構造器。如下:
public class Test {
public static void main(String[] args) {
MyThread myThread = new MyThread();
Thread t = new Thread(myThread);
t.start();
}
static class MyThread implements Runnable {
@Override
public void run() {
System.out.println("執行緒:" + Thread.currentThread().getName());
}
}
}
可以看到使用匿名內部類的話就省略了新建Runnable介面的實現類這一步驟。
3. 使用Lambda表示式
上面使用匿名內部類的寫法,如果使用Lambda表示式可以寫成下面這樣:
public class Test {
public static void main(String[] args) {
Thread t = new Thread(() -> {
System.out.println("執行緒:" + Thread.currentThread().getName());
});
t.start();
}
}
這樣有一個問題,如果介面裡面有多個方法,那麼Lambda表示式怎麼知道實現的是哪個方法呢?我們通過程式碼測試一下:
package com.wangjun.othersOfJava;
public class LambdaTest {
public static void main(String[] args) {
Animal a = () -> { // 編譯報錯:The target type of this expression must be a functional interface
System.out.println("狗狗吃飯");
};
a.eat();
}
interface Animal {
public void eat();
public void duty();
}
}
可以看到編譯報錯,這個提到一個functional interface,就是函式式介面。函式式介面就是隻有一個抽象方法的介面。這樣,就不難理解了,原來Lambda表示式只支援函式式介面。
4. Lambda表示式使用的幾種方式
package com.wangjun.othersOfJava;
public class LambdaTest {
public static void main(String[] args) {
// 帶型別
Animal a1 = (String str) -> {
System.out.println("狗狗吃飯:" + str);
};
// 不帶型別
Animal a2 = (str) -> {
System.out.println("狗狗吃飯:" + str);
};
// 不帶括號
Animal a3 = str -> {
System.out.println("狗狗吃飯:" + str);
};
// 不帶大括號
Animal a4 = str -> System.out.println("狗狗吃飯:" + str);
a1.eat("火腿腸");
a2.eat("牛肉");
a3.eat("麵條");
a4.eat("米飯");
// 使用return返回
Person p1 = () -> {
return "老師的職責:教書育人!";
};
// 直接返回
Person p2 = () -> "醫生的職責:救死扶傷!";
System.out.println(p1.duty());
System.out.println(p2.duty());
}
// 沒有返回值
interface Animal {
public void eat(String str);
}
// 有返回值
interface Person {
public String duty();
}
}
5. Java的雙冒號表示式
JDK8中有雙冒號的用法,就是把方法當做引數傳到stream內部,使stream的每個元素都傳入到該方法裡面執行一下。下面通過遍歷一個List來說明一下雙冒號和Lambda表示式使用方式的不同。
package com.wangjun.othersOfJava;
import java.util.Arrays;
import java.util.List;
import java.util.function.Consumer;
public class LambdaTest {
public static void printStr(String str) {
System.out.println(str);
}
public static void main(String[] args) {
List<String> list = Arrays.asList("aaa","bbb","ccc");
// 1.通常的遍歷方式
for(String str: list) {
LambdaTest.printStr(str);
}
// 2.使用Lambda表示式遍歷
list.forEach(str -> {
LambdaTest.printStr(str);
});
// 3.使用::遍歷
list.forEach(LambdaTest::printStr);
// 下面的方法和上面等價,使用的是函數語言程式設計
Consumer<String> methodParam = LambdaTest::printStr; //方法引數
list.forEach(x -> methodParam.accept(x));//方法執行accept
}
}