1.Lambda 表達式,也可稱爲閉包,它是推動 Java 8 發佈的最重要新特性。
Lambda 允許把函數作爲一個方法的參數(函數作爲參數傳遞進方法中)。
使用 Lambda 表達式可以使代碼變的更加簡潔緊湊。
2.以下是lambda表達式的重要特徵:
- 可選類型聲明:不需要聲明參數類型,編譯器可以統一識別參數值。
- 可選的參數圓括號:一個參數無需定義圓括號,但多個參數需要定義圓括號。
- 可選的大括號:如果主體包含了一個語句,就不需要使用大括號。
- 可選的返回關鍵字:如果主體只有一個表達式返回值則編譯器會自動返回值,大括號需要指定明表達式返回了一個數值。
3.Lambda 表達式的簡單例子
// 1. 不需要參數,返回值爲 5
() -> 5
// 2. 接收一個參數(數字類型),返回其2倍的值
x -> 2 * x
// 3. 接受2個參數(數字),並返回他們的差值
(x, y) -> x – y
// 4. 接收2個int型整數,返回他們的和
(int x, int y) -> x + y
// 5. 接受一個 string 對象,並在控制檯打印,不返回任何值(看起來像是返回void)
(String s) -> System.out.print(s)
4.樣例代碼如下:
public class Java8LambdaTest {
@FunctionalInterface
interface T1 {
int add(int a, int b);
}
@FunctionalInterface
interface T2 {
String call(String message);
// 添加了 @FunctionalInterface 註解的接口值允許有一個待實現的方法
// 否則編譯不過去
//String call2(String message);
}
private String operate(int a,int b,T1 t){
return "result:"+t.add(a,b);
}
public static void main(String[] args) {
Java8LambdaTest bean = new Java8LambdaTest();
// 類型聲明(類似於之前的內部匿名類的實現)
T1 add = (int a, int b) -> a + b;
// 不用類型聲明(類似於之前的內部匿名類的實現)
T1 sub = (a, b) -> a - b;
// 大括號中的返回語句(類似於之前的內部匿名類的實現)
T1 multiplication = (int a, int b) -> { return a * b; };
// 沒有大括號及返回語句(類似於之前的內部匿名類的實現)
T1 division = (int a, int b) -> a / b;
System.out.println("10 + 5 = " + bean.operate(10, 5, add));
System.out.println("10 - 5 = " + bean.operate(10, 5, sub));
System.out.println("10 x 5 = " + bean.operate(10, 5, multiplication));
System.out.println("10 / 5 = " + bean.operate(10, 5, division));
// 不用括號(一個輸入參數一個返回值,太方便簡單了)
T2 service1 = message -> "Hello1 " + message;
// 用括號
T2 service2 = (message) -> "Hello2 " + message;
System.out.println(service1.call("Tom1"));
System.out.println(service2.call("Tom2"));
}
}
輸出結果:
10 + 5 = result:15
10 - 5 = result:5
10 x 5 = result:50
10 / 5 = result:2
Hello1 Tom1
Hello2 Tom2
5.使用 Lambda 表達式需要注意以下兩點:
- Lambda 表達式主要用來定義行內執行的方法類型接口,例如,一個簡單方法接口。在上面例子中,我們使用各種類型的Lambda表達式來定義 T1 接口的方法。然後我們定義了 call 的執行。
- Lambda 表達式免去了使用匿名方法的麻煩,並且給予Java簡單但是強大的函數化的編程能力。
6.變量作用域
lambda 表達式只能引用標記了 final 的外層局部變量,這就是說不能在 lambda 內部修改定義在域外的局部變量,
否則會編譯錯誤。
public class Java8Tester {
final static String salutation = "Hello! ";
public static void main(String args[]){
GreetingService greetService1 = message ->
System.out.println(salutation + message);
greetService1.sayMessage("Runoob");
}
interface GreetingService {
void sayMessage(String message);
}
}
我們也可以直接在 lambda 表達式中訪問外層的局部變量:
public class Java8Tester {
public static void main(String args[]) {
final int num = 1;
Converter<Integer, String> s = (param) ->
System.out.println(String.valueOf(param + num));
s.convert(2); // 輸出結果爲 3
}
public interface Converter<T1, T2> {
void convert(int i);
}
}
lambda 表達式的局部變量可以不用聲明爲 final,但是必須不可被後面的代碼修改(即隱性的具有 final 的語義)
int num = 1;
Converter<Integer, String> s = (param) -> System.out.println(String.valueOf(param + num));
s.convert(2);
num = 5;
//報錯信息:Local variable num defined in an enclosing scope must be final or effectively final
在 Lambda 表達式當中不允許聲明一個與局部變量同名的參數或者局部變量。
String first = "";
Comparator<String> comparator = (first, second) -> Integer.compare(first.length(), second.length()); //編譯會出錯
7.把方法的所有參數全部傳遞給引用的構造器,根據參數類型自動推斷調用的構造器方法;
public class Java8Test1 {
@FunctionalInterface
interface CreateString{
String create(char[] chars);
}
public static void main(String[] args) {
// 引用構造器方法
CreateString test = String::new;
System.out.println(test.create(new char[]{'j','m','m'}));
}
}
以上學習筆記參考:http://www.runoob.com/java/java8-lambda-expressions.html