成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

詳解 Java 中的 Lambda

開發 后端
如果你想給一個 Java 變量賦一段“代碼”,該怎么辦呢?在 Java 8 之前,這是不可能的。但在 Java 8 出現后,可以使用 Lambda 特性來做到這一點。

什么是 Lambda

我們知道,對于一個 Java 變量,我們可以給它賦一個“值”,然后可以用它做一些操作。

Integer a = 1;
String s = "Hello";
System.out.println(s + a);

如果你想給一個 Java 變量賦一段“代碼”,該怎么辦呢?例如,我想把右邊的代碼塊賦給一個名為 codeBlock 的 Java 變量。

在 Java 8 之前,這是不可能的。但在 Java 8 出現后,可以使用 Lambda 特性來做到這一點。

以下就是最直觀的寫法:

實際上是不允許這樣寫的會編譯失敗,這里只是為了讓大家方便理解

codeBlock = public void doSomething(String s) {
    System.out.println(s);
}

這種寫法不是很簡潔。我們可以去掉一些無用的聲明對代碼進行簡化。

codeBlock = public void doSomething(String s) {
   System.out.println(s);
}
// 這里的 public 是多余的,因為在這個上下文中不需要訪問修飾符。
codeBlock = void doSomething(String s) {
   System.out.println(s);
}
// 函數名 doSomething 也是多余的,因為已經將函數體賦值給了 codeBlock。
codeBlock = void (String s) {
   System.out.println(s);
}
// 編譯器可以自行推斷返回類型,這里不需要顯式地寫出 void。
codeBlock = (String s) {
   System.out.println(s);
}
// 編譯器可以自行推斷輸入參數類型,這里不需要顯式地寫出 String 類型。
codeBlock = (s) -> System.out.println(s);

這樣,我們就將一段“代碼”賦給了一個變量。而“這段代碼”,或者說“賦給變量的這個函數”,就是一個 Lambda 表達式。

但這里還有一個問題,即變量 codeBlock 應該是什么類型呢?在 Java 8 中,所有 Lambda 類型都是一個接口,而 Lambda 表達式本身,也就是“這段代碼”,需要是這個接口的一個實現。在我看來,這是理解 Lambda 的關鍵。簡而言之,Lambda 表達式本身就是一個接口的實現。直接這么說可能還是有點讓人困惑,所以我們繼續舉例。我們給上面的 codeBlock 添加一個類型:

codeBlock = (s)->System.out.println(s);

interface LambdaInterface {
    public void doSomething(String s);
}

這種只有一個函數需要實現的接口稱為“函數式接口”。為了防止后來的人給這個接口添加接口函數,導致有多個接口函數需要實現而變成“非函數式接口”,我們可以給這個接口添加一個聲明@FunctionalInterface,這樣其他人就不能給它添加新函數了。

@FunctionalInterface
interface LambdaInterface {
    public void doSomething(String s);
}

這樣,我們就得到了一個完整的 Lambda 表達式聲明。

LambdaInterface codeBlock =(s)System.out.println(s);

Lambda 表達式的作用是什么

最直觀的作用就是使代碼極其簡潔。我們可以比較一下 Lambda 表達式和傳統 Java 對同一接口的實現:

interface LambdaInterface {
public void doSomething(String s);
}

// Java 8
LambdaInterface codeBlock = (s) -> System.out.println(s);

// Java 7
publicclass LambdaInterfaceImpl implements LambdaInterface {
@Override
public void doSomething(String s) {
   System.out.println(s);
 }
}

這兩種寫法本質上是等價的。但顯然,Java 8 中的寫法更優雅簡潔。而且,由于 Lambda 可以直接賦給變量,我們可以直接將 Lambda 作為參數傳遞給函數,而 java7 必須有明確的接口實現和初始化定義:

// 定義了一個靜態方法 useLambda,它接受一個 LambdaInterface 類型的參數和一個 String 類型的參數。
public static void useLambda(LambdaInterface lambdaInterface, String s) {
    lambdaInterface.doSomething(s);
}

// Java 8
// 直接使用 Lambda 表達式調用 useLambda 方法。
   useLambda(s -> System.out.println(s), "Hello");
// Java 7
// 定義了一個 LambdaInterface 接口和一個實現該接口的 LambdaInterfaceImpl 類。
   interface LambdaInterface {
    public void doSomething(String s);
}

publicclass LambdaInterfaceImpl implements LambdaInterface {
    @Override
    public void doSomething(String s) {
        System.out.println(s);
    }
}
// 實例化 LambdaInterfaceImpl 類,并將實例傳遞給 useLambda 方法。
LambdaInterface myLambdaInterface = new LambdaInterfaceImpl();
useLambda(myLambdaInterface, "Hello");

在某些情況下,這個接口實現只需要使用一次。Java 7 要求你定義一個接口然后實現它。相比之下,Java 8 的 Lambda 看起來干凈得多。Lambda 結合了函數式接口庫、forEach、stream()、方法引用等新特性,使代碼更加簡潔!我們直接看例子。

@Getter
@AllArgsConstructor
public static class Student {
    private String name;
    private Integer age;
}

List<Student> students = Arrays.asList(
        new Student("Bob", 18),
        new Student("Ted", 17),
        new Student("Zeka", 18)
);

現在你需要打印出 students 中所有 18 歲學生的名字。

原始的 Lambda 寫法:定義兩個函數式接口,定義一個靜態函數,調用靜態函數并給參數賦值 Lambda 表達式。

@FunctionalInterface
interface AgeMatcher {
    boolean match(Student student);
}

@FunctionalInterface
interface Executor {
    boolean execute(Student student);
}

public static void matchAndExecute(List<Student> students, AgeMatcher matcher, Executor executor) {
    for (Student student : students) {
        if (matcher.match(student)) {
            executor.execute(student);
        }
    }
}

public static void main(String[] args) {
    List<Student> students = Arrays.asList(
            new Student("Bob", 18),
            new Student("Ted", 17),
            new Student("zeka", 18)
    );
    matchAndExecute(students,
            s -> s.getAge() == 18,
            s -> System.out.println(s.getName())
    );
}

這段代碼實際上已經比較簡潔了,但我們還能更簡潔嗎?當然可以,Java 8 中有一個函數式接口包,它定義了大量可能用到的函數式接口(java.util.function (Java Platform SE 8))。

因此,我們根本不需要在這里定義 AgeMatcher 和 Executor 這兩個函數式接口。我們可以直接使用 Java 8 函數式接口包中的 Predicate(T) 和 Consumer(T),因為它們的一對接口定義實際上與 AgeMatcher/Executor 相同。

第一步簡化:利用函數式接口

public static void matchAndExecute(List<Student> students, Predicate<Student> predicate, Consumer<Student> consumer) {
    for (Student student : students) {
        if (predicate.test(student)) {
            consumer.accept(student);
        }
    }
}

matchAndExecute 中的 forEach 循環實際上很煩人。這里可以使用 Iterable 自帶的 forEach 代替。forEach 本身可以接受一個 Consumer(T) 參數。

第二步簡化:用 Iterable.forEach 代替 forEach 循環:

public static void matchAndExecute(List<Student> students, Predicate<Student> predicate, Consumer<Student> consumer) {
    students.forEach(s -> {
        if (predicate.test(s)) {
            consumer.accept(s);
        }
    });
}

由于 matchAndExecute 實際上只是對 List 的一個操作,這里我們可以去掉 matchAndExecute,直接使用 stream() 特性來完成它。stream() 的幾個方法接受 Predicate(T) 和 Consumer(T) 等參數(java.util.stream (Java Platform SE 8))。一旦你理解了上面的內容,stream() 就很容易理解,不需要進一步解釋。

第三步簡化:用 stream() 代替靜態函數:

students.stream()
       .filter(s -> s.getAge() == 18)
       .forEach(s -> System.out.println(s.getName()));

與最初的 Lambda 寫法相比代碼量已經減少了非常多。但如果我們要求改為打印學生的所有信息,并且s -> System.out.println(s);那么我們可以使用方法引用來繼續簡化。所謂方法引用,就是用已經編寫好的其他 Object/Class 的方法來代替 Lambda 表達式。格式如下:

第四步簡化:可以在 forEach 中使用方法引用代替 Lambda 表達式:

students.stream()
       .filter(s -> s.getAge() == 18)
       .map(Student::getName)
       .forEach(System.out::println);

這基本上是我能寫出的最簡潔的版本了。

關于 Java 中的 Lambda 還有一些需要討論和學習的地方。例如,如何利用 Lambda 的特性進行并行處理等??傊抑皇墙o你一個大致的介紹,讓你有個概念。網上有很多關于 Lambda 的相關教程,多讀多練,隨著時間的推移肯定能夠掌握它。

責任編輯:趙寧寧 來源: 程序猿技術充電站
相關推薦

2024-03-12 08:23:54

JavaLambda函數式編程

2009-07-01 09:56:10

C#3.0

2019-10-10 17:53:36

大數據平臺架構LambdaKappa

2023-05-31 13:32:08

Javalambda函數

2021-01-21 05:46:22

JavaLambda開發

2009-06-22 10:34:43

Boost庫lambda

2009-09-14 13:57:20

C# Lambda表達Lambda表達式

2020-09-26 07:19:46

Java

2016-09-18 16:58:09

JavaProperties

2013-01-05 02:19:50

JavaLambda表達式JVM

2012-07-18 09:45:32

Java 8ScalaLambda

2023-10-10 10:43:19

JavaJDK1.8

2009-11-12 10:55:17

Lambda表達式

2020-03-29 20:38:35

PythonLambda語言

2009-06-25 15:20:28

CollectionMap

2025-02-05 12:22:21

2009-06-30 15:18:10

StringBuildJava

2012-06-26 10:03:58

JavaJava 8lambda

2009-07-09 09:51:07

Lambda表達式C#

2022-06-27 08:36:08

PythonLambda
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 午夜影院在线观看 | 成人aaa视频 | 国产一区免费 | 国产目拍亚洲精品99久久精品 | 国产亚洲精品美女久久久久久久久久 | 国产免费一区二区三区 | 欧美成人免费电影 | 欧美精品久久久久久久久久 | 欧美一区二区在线观看 | 日日操夜夜操天天操 | 2022精品国偷自产免费观看 | 成av在线| av网站在线看 | 精品国产免费一区二区三区五区 | 四季久久免费一区二区三区四区 | 日本成人一区二区 | av男人的天堂在线 | 丁香婷婷综合激情五月色 | av中文字幕在线播放 | 99久久婷婷国产综合精品电影 | 欧美九九九 | 亚洲成人国产 | 成人福利视频网站 | 男女精品网站 | 国产色| 性高湖久久久久久久久aaaaa | 精品久久久久久久久久久久 | 国产视频精品在线 | 欧美国产视频一区二区 | 中文字幕在线视频网站 | 亚洲喷水 | 欧美在线一区二区三区 | 成人精品视频99在线观看免费 | 狠狠干美女 | 国产在线观看一区二区三区 | 午夜欧美 | 日韩av免费看| 艹逼网 | 日韩第一页 | 中文字幕久久精品 | 国产 欧美 日韩 一区 |