Lambda用法

Lambda 表達式,也可稱為閉包,它是推動 Java 8 發(fā)布的最重要新特性。

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

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

語法

(Type1 param1, Type2 param2, ..., TypeN paramN) -> {
  statement1;
  statement2;
  //.............
  return statementM;
}

lambda表達式的重要特征:

  • 可選類型聲明:不需要聲明參數(shù)類型,編譯器可以統(tǒng)一識別參數(shù)值。
  • 可選的參數(shù)圓括號:一個參數(shù)無需定義圓括號,但多個參數(shù)或沒有參數(shù)需要定義圓括號。
  • 可選的大括號:如果主體包含了一個語句,就不需要使用大括號。
  • 可選的返回關(guān)鍵字:如果主體只有一個表達式返回值則編譯器會自動返回值,大括號需要指定明表達式返回了一個數(shù)值。

簡單例子:

// 1. 不需要參數(shù),返回值為 5  
() -> 5  
  
// 2. 接收一個參數(shù)(數(shù)字類型),返回其2倍的值  
x -> 2 * x  
  
// 3. 接受2個參數(shù)(數(shù)字),并返回他們的差值  
(x, y) -> x – y  
  
// 4. 接收2個int型整數(shù),返回他們的和  
(int x, int y) -> x + y  
  
// 5. 接受一個 string 對象,并在控制臺打印,不返回任何值(看起來像是返回void)  
(String s) -> System.out.print(s)

對于Lambda語法的簡潔緊湊,我們可以和之前的常用語法做個對比:

  • 不使用Lambda
public class Test {

    public static void main(String[] args) {
        Integer[] array = {1, 2, 3, 4, 5, 6, 7};
        List<Integer> arrayList = Arrays.asList(array);
        System.out.println("排序前:");
//        遍歷集合,我們常用的方法有for循環(huán),增強的for循環(huán),迭代器遍歷
        for (Integer number : arrayList) {
            System.out.print(number + " ");
        }
//        將集合內(nèi)數(shù)字降序排序
        arrayList.sort(new Comparator<Integer>() {
            @Override
            public int compare(Integer o1, Integer o2) {
                return o2.compareTo(o1);
            }
        });
        System.out.println("\n排序后:");
//        遍歷集合
        for (Integer number : arrayList) {
            System.out.print(number + " ");
        }
    }
}
  • 使用Lambda
public class Test {

    public static void main(String[] args) {
        Integer[] array = {1, 2, 3, 4, 5, 6, 7};
        List<Integer> arrayList = Arrays.asList(array);
        System.out.println("排序前:");
//        遍歷集合
        arrayList.forEach(x -> System.out.print(x + " "));
//        將集合內(nèi)數(shù)字降序排序
        arrayList.sort((o1, o2) -> o2.compareTo(o1));
        System.out.println("\n排序后:");
//        遍歷集合
        arrayList.forEach(x -> System.out.print(x + " "));
    }
}

可以很直觀地看到,Lambda語法更加簡潔緊湊,尤其是對于上例匿名方法Comparator的使用中可以體現(xiàn)這一點。

輸出均為:

排序前:
1 2 3 4 5 6 7 
排序后:
7 6 5 4 3 2 1 

變量作用域

lambda 表達式只能引用標記了 final 的外層局部變量,這就是說不能在 lambda 內(nèi)部修改定義在域外的局部變量,否則會編譯錯誤。

public class Test {

    public static void main(String[] args) {
        Integer[] array = {1, 2, 3, 4, 5, 6, 7};
        List<Integer> arrayList = Arrays.asList(array);
        final int DOUBLE = 2;
        arrayList.forEach(x -> System.out.println(x * DOUBLE));
    }
}

輸出:

2
4
6
8
10
12
14

lambda 表達式的局部變量可以不用聲明為 final,但是必須不可被后面的代碼修改(即隱性的具有 final 的語義)

  • 隱性的具有 final 的語義

    public static void main(String[] args) {
        Integer[] array = {1, 2, 3, 4, 5, 6, 7};
        List<Integer> arrayList = Arrays.asList(array);
        int DOUBLE = 2;
        arrayList.forEach(x -> System.out.println(x * DOUBLE));
    }
}

輸出:

2
4
6
8
10
12
14
  • final
public class Test {

    public static void main(String[] args) {
        Integer[] array = {1, 2, 3, 4, 5, 6, 7};
        List<Integer> arrayList = Arrays.asList(array);
        int DOUBLE = 2;
        arrayList.forEach(x -> System.out.println(x * DOUBLE));
        DOUBLE = 3;
    }
}

編譯時會報錯:從lambda 表達式引用的本地變量必須是最終變量或?qū)嶋H上的最終變量。

另外,在 Lambda 表達式當中不允許聲明一個與局部變量同名的參數(shù)或者局部變量。

public class Test {

    public static void main(String[] args) {
        Integer[] array = {1, 2, 3, 4, 5, 6, 7};
        List<Integer> arrayList = Arrays.asList(array);
        int first = 1;
        arrayList.sort((first, second) -> first.compareTo(second));
        arrayList.forEach(x -> System.out.println(x));
    }
}

編譯時會報錯:已在方法 main(java.lang.String[])中定義了變量 first

方法引用

靜態(tài)方法或?qū)嵗椒ǖ囊谜Z法:

Class or instance :: method

則上面例子中的遍歷并輸出集合元素的代碼,也可以寫為:

public class Test {

    public static void main(String[] args) {
        Integer[] array = {1, 2, 3, 4, 5, 6, 7};
        List<Integer> arrayList = Arrays.asList(array);
        arrayList.forEach(System.out::println);
    }
}

構(gòu)造器引用

語法:

ClassName::new

例子

有一個Cat類:

public class Cat {
    private String name;

    public Cat(String name) {
        this.name = name;
    }

    public String getName() {
        return this.name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

有一個nameList,遍歷nameList,調(diào)用Cat構(gòu)造函數(shù),生成一個catList

import java.util.*;
import java.util.function.Function;

public class Test {

    public static void main(String[] args) {
        String[] array = {"tay", "mere", "amy"};
        List<String> nameList = Arrays.asList(array);
        List<Cat> cats = getCatList(nameList, Cat::new);
        cats.forEach(cat -> System.out.println(cat.getName()));
    }

    public static List<Cat> getCatList(List<String> nameList, Function<String, Cat> function) {
        List<Cat> result = new ArrayList<>();
        nameList.forEach(name -> result.add(function.apply(name)));
        return result;
    }
}

輸出:

tay
mere
amy
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容