Java8語法糖之Lambda表達式的深入講解
一、Lambda表達式簡介
Lambda表達式,是Java8的一個新特性,也是Java8中最值得學習的新特性之一。(另一個新特性是流式編程。)
Lambda表達式,從本質上講是一個匿名方法。可以使用這個匿名方法,實現接口中的方法。
功能:通常使用Lambda表達式,是為瞭簡化接口實現的。關於接口實現可以有多種方式實現,例如:①設計接口的實現類、②使用匿名內部類。但是③使用lambda表達式,比這兩種方式都簡單。
要求:lambda表達式,隻能實現函數式接口:即一個接口中,要求實現類必須實現的抽象方法,有且隻有一個。
@FunctionalInterface註解 ,用在接口之前,用來判斷接口是否是一個函數式接口。如果不是函數式接口會報錯。功能類似於@Override。
二、Lambda表達式語法
lambda表達式本質上是一個匿名方法,因此再寫lambda表達式時,不需要關心方法名是什麼,也不需要關心返回值類型。隻需要關心兩部分:參數列表、方法體。
()參數部分:方法的參數列表,要求和實現的接口中的方法參數部分一致,包括參數的數量和類型。
{}方法體部分:方法的實現部分,如果接口中定義的方法有返回值,則在實現時,註意返回值的返回。
-> :分隔參數部分和方法體部分。
Lambda表達式基礎語法:
(參數) ->{
方法體
}
下面定義6種參數和返回值各不相同的函數式接口,分別使用lambda表達式對接口中的方法進行實現:
下面是針對上面6種函數式接口的lambda表達式實現。
/** * @Description: * @author Guoqianliang * @date 19:50 - 2021/2/15 */ public class BasicSyntax { public static void main(String[] args) { // 1.實現無參數,無返回值的函數式接口 NoneReturnNoneParameter lambda1 = () -> { System.out.println("這是無參,無返回值的方法"); }; lambda1.test(); // 2.實現一個參數,無返回值的函數式接口 NoneReturnSingleParameter lambda2 = (int a) -> { System.out.println("這是一個參數,無返回值的方法,參數a:" + a); }; lambda2.test(10); // 3.實現多個參數,無返回值的函數式接口 NoneReturnMutipleParameter lambda3 = (int a, int b) -> { System.out.println("這是多個參數,無返回值的方法,參數a=" + a + ",b=" + b); }; lambda3.test(10, 20); // 4.實現無參數,有返回值有返回值的函數式接口 SingleReturnNoneParameter lambda4 = () -> { System.out.println("這是無參數,有返回值的方法,返回值是:"); return 10; }; System.out.println(lambda4.test()); // 5.實現一個參數,有返回值的函數式接口 SingleReturnSingleParameter lambda5 = (int a) -> { System.out.println("這是一個參數,有返回值的方法,返回值是:"); return a; }; System.out.println(lambda5.test(10)); // 6.實現多個參數,有返回值的函數式接口 SingleReturnMutipleParameter lambda6 = (int a, int b) -> { System.out.println("這是多個參數,有返回值的方法,返回值是:"); return a + b; }; System.out.println(lambda6.test(1, 2)); } }
語法精簡進階:
- 參數列表的參數類型可以省略。
- 如果參數列表中的參數有且隻有一個,可以省略小括號。
- 如果方法體中隻有一條語句,可以省略大括號。(註:如果這條語句是返回語句,省略瞭大括號後也要把return關鍵字省略)
三、函數引用
lambda表達式是為瞭簡化接口。在lambda表達式中,不應該出現比較復雜的邏輯。如果需要處理的邏輯比較復雜,一般情況會單獨寫一個方法。在lambda表達式中直接引用這個方法即可。即引用一個已經存在的方法,使其代替lambda表達式完成接口的實現。
1.靜態方法引用
語法:類::靜態方法
在引用的方法後面,不要添加小括號。
引用的這個方法,參數(數量、類型)和返回值,必須要跟接口中定義的一致。
/** * @Description: 方法引用 * @author Guoqianliang * @date 0:26 - 2021/2/16 */ public class Lambda1 { private static interface Calculate { int calculate(int a, int b); } private static int calculate(int x, int y) { if (x > y) { return x - y; } else if (x < y) { return y - x; } return x + y; } public static void main(String[] args) { // 靜態方法引用 Calculate calculate = Lambda1::calculate; System.out.println(calculate.calculate(10, 20)); } }
2.非靜態方法引用
語法:對象::非靜態方法
在引用的方法後面,不要添加小括號。
引用的這個方法,參數(數量、類型)和返回值,必須要跟接口中定義的一致。
/** * @Description: 方法引用 * @author Guoqianliang * @date 0:26 - 2021/2/16 */ public class Lambda1 { private static interface Calculate { int calculate(int a, int b); } // 非靜態方法 private int calculate2(int a, int b) { if (a != b) { return a - b; } return a + b; } public static void main(String[] args) { // 非靜態方法引用 Calculate calculate2 = new Lambda1()::calculate2; System.out.println(calculate.calculate(10, 20)); } }
3.構造方法引用
語法:類名::new
可以通過接口中的方法的參數,區分引用不同的構造方法。
如果某一個函數式接口中定義的方法,僅僅是為瞭得到一個類的對象。此時就可以使用構造方法的引用,簡化這個方法的實現。
/** * @Description: 構造方法引用 * @author Guoqianliang * @date 11:20 - 2021/2/16 */ public class Lambda2 { @FunctionalInterface private interface GetPersonWithNoneParameter { Person get(); } @FunctionalInterface private interface GetPersonWithSingleParameter { Person get(String name); } @FunctionalInterface private interface GetPersonWithMutipleParameter { Person get(String name, int age); } private static class Person { String name; int age; public Person() { System.out.println("Person類的無參構造方法執行瞭"); } public Person(String name) { this.name = name; System.out.println("Person類的有參構造方法執行瞭"); } public Person(String name, int age) { this.name = name; this.age = age; System.out.println("Person類的兩個參數的構造方法執行瞭"); } } public static void main(String[] args) { // 1.使用lambda表達式,實現GetPersonWithNoneParameter接口 GetPersonWithNoneParameter getPerson = Person::new; // 2.使用lambda表達式,實現GetPersonWithSingleParameter接口 GetPersonWithSingleParameter getPerson2 = Person::new; // 3.使用lambda表達式,實現GetPersonWithMutipleParameter接口 GetPersonWithMutipleParameter getPerson3 = Person::new; System.out.println(getPerson.get()); System.out.println(getPerson2.get("樹先生")); System.out.println(getPerson3.get("你好", 23)); } }
4.對象方法的特殊引用
使用lambda表達式實現某些接口時,如果lambda表達式中包含瞭某一個對象,此時方法體中,直接使用這個對象調用它的某一個方法就可以完成整體的邏輯。
/** * @Description: 對象方法的特殊應用 * @author Guoqianliang * @date 11:54 - 2021/2/16 */ public class Lambda3 { @FunctionalInterface private interface MyInterface { // String get(Person person); void set(Person person, String name); } private static class Person { private String name; public void setName(String name) { this.name = name; } public String getName() { return name; } } public static void main(String[] args) { Person p1 = new Person(); p1.setName("小明"); // 邏輯實現隻是為瞭獲取到對象的名字 // MyInterface lambda2 = Person::getName; // System.out.println(lambda2.get(p1)); // 邏輯實現隻是為瞭給對象的某些屬性進行賦值 MyInterface lambda1 = (x, n) -> x.setName(n); MyInterface lambda2 = Person::setName; lambda2.set(p1, "李華"); System.out.println(p1.getName()); } }
四、Lambda表達式需要註意的問題
如果用到局部變量,默認會被聲明為常量,不能發生值的改變。
/** * @Description: * @author Guoqianliang * @date 13:05 - 2021/2/16 */ public class Lambda4 { public static void main(String[] args) { // 1.定義一個局部變量 int x = 10; // 2.使用lambda表達式實現接口 LambdaTest lambda = () -> { System.out.println("x=" + x); }; // 3. 無法修改常量x // x=20; } } @FunctionalInterface interface LambdaTest { void test(); }
總結
到此這篇關於Java8語法糖之Lambda表達式的文章就介紹到這瞭,更多相關Java8語法糖Lambda表達式內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- Java方法引用原理實例解析
- 關於JavaEE匿名內部類和Lambda表達式的註意事項
- Lambda表達式原理及示例
- java Lambda表達式的使用心得
- 簡單易懂的java8新特性之lambda表達式知識總結