天天看點

Java Lambda 表達式

Java 8 新特性

Lambda 表達式,也可稱為閉包,它是推動 Java 8 釋出的最重要新特性。

Lambda 允許把函數作為一個方法的參數(函數作為參數傳遞進方法中)。

使用 Lambda 表達式可以使代碼變的更加簡潔緊湊。

文法

lambda 表達式的文法格式如下:

(parameters) -> expression

(parameters) ->{ statements; }

以下是lambda表達式的重要特征:

  • 可選類型聲明: 不需要聲明參數類型,編譯器可以統一識别參數值。
  • 可選的參數圓括号: 一個參數無需定義圓括号,但多個參數需要定義圓括号。
  • 可選的大括号: 如果主體包含了一個語句,就不需要使用大括号。
  • 可選的傳回關鍵字: 如果主體隻有一個表達式傳回值則編譯器會自動傳回值,大括号需要指定表達式傳回了一個數值。

Lambda 表達式執行個體

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)      

在 Java8Tester.java 檔案輸入以下代碼:

Java8Tester.java 檔案

public class Java8Tester {

public static void main(String args[]){

Java8Tester tester = new Java8Tester();

// 類型聲明

MathOperation addition = (int a, int b) -> a + b;

// 不用類型聲明

MathOperation subtraction = (a, b) -> a - b;

// 大括号中的傳回語句

MathOperation multiplication = (int a, int b) -> { return a * b; };

// 沒有大括号及傳回語句

MathOperation division = (int a, int b) -> a / b;

System.out.println("10 + 5 = " + tester.operate(10, 5, addition));

System.out.println("10 - 5 = " + tester.operate(10, 5, subtraction));

System.out.println("10 x 5 = " + tester.operate(10, 5, multiplication));

System.out.println("10 / 5 = " + tester.operate(10, 5, division));

// 不用括号

GreetingService greetService1 = message ->

System.out.println("Hello " + message);

// 用括号

GreetingService greetService2 = (message) ->

greetService1.sayMessage("Runoob");

greetService2.sayMessage("Google");

}

interface MathOperation {

int operation(int a, int b);

interface GreetingService {

void sayMessage(String message);

private int operate(int a, int b, MathOperation mathOperation){

return mathOperation.operation(a, b);

}

執行以上腳本,輸出結果為:

$ javac Java8Tester.java 
$ java Java8Tester
10 + 5 = 15
10 - 5 = 5
10 x 5 = 50
10 / 5 = 2
Hello Runoob
Hello Google
      

使用 Lambda 表達式需要注意以下兩點:

  • Lambda 表達式主要用來定義行内執行的方法類型接口,例如,一個簡單方法接口。在上面例子中,我們使用各種類型的Lambda表達式來定義MathOperation接口的方法。然後我們定義了sayMessage的執行。
  • Lambda 表達式免去了使用匿名方法的麻煩,并且給予Java簡單但是強大的函數化的程式設計能力。

變量作用域

lambda 表達式隻能引用标記了 final 的外層局部變量,這就是說不能在 lambda 内部修改定義在域外的局部變量,否則會編譯錯誤。

final static String salutation = "Hello! ";

GreetingService greetService1 = message ->

System.out.println(salutation + message);

$ javac Java8Tester.java 
$ java Java8Tester
Hello! Runoob
      

我們也可以直接在 lambda 表達式中通路外層的局部變量:

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());  //編譯會出錯