Java特性?Lambda?表達式和函數(shù)式接口
Java Lambda表達式
為方便使用Java函數(shù)式接口,一定要搞清楚Java的Lambda表達式怎么書寫。同時Java的Lambda不是那么通俗易懂,所以一定得學(xué)明白,不然其實后面涉及到Lambda的代碼部分會變得晦澀難懂。而且掌握Lambda表達式可以讓你寫出別人不容易看懂的Java代碼,比較cool。
起源&概念
2014年3月Oracle所發(fā)布的Java 8(JDK 1.8)中,加入了Lambda表達式的重量級新特性,為我們打開了新世界的大門。
Lambda表達式基于函數(shù)式編程思想,在數(shù)學(xué)中函數(shù)就是有輸入量、輸出量的一套計算方案,也就是“拿什么東西做什么事”。和我們之前學(xué)的面向?qū)ο蟮木幊趟枷?ldquo;通過對象來做事情”有很大不同。函數(shù)式編程思想盡可能簡化和忽略了面向?qū)ο蟮膹?fù)雜語法----強調(diào)做什么,而不是像面向?qū)ο竽菢訌娬{(diào)以什么形式做什么。
Java Lambda 表達式,也可以稱為閉包,是Java 8引入的重要新特性, Lambda允許把函數(shù)作為一個方法的參數(shù),使用它可以使代碼變得簡潔緊湊。
語法
- ( ) -> { }
- 一個放參數(shù)的圓括號 ():里面放參數(shù),也可以沒有,多個就以 ' , ' 分割
- 一個箭頭 -> :用來傳遞參數(shù)到方法體中
- 一些要執(zhí)行的代碼 { } :重寫的接口中的抽象方法的方法體
(parameters)->expression //或 (parameter)->{statements;}
下面是lambda表達式的一些重要特征:
- 可選類型聲明:不需要聲明參數(shù)類型,編譯器可以統(tǒng)一識別參數(shù)值。
- 可選的參數(shù)圓括號:一個參數(shù)無需定義圓括號,但多個參數(shù)需要。
- 可選的大括號:如果主題包含了一個語句,就不需要使用大括號。
- 可選的返回關(guān)鍵字:如果主體只有一個表達式返回值,則編譯器會自動返回值,大括號需要指定明表達式返回了一個值。
簡單例子
//01 不需要參數(shù),返回值是5。 ()->5 //02 接收一個參數(shù)(數(shù)字類型),返回其兩倍的值。 (x)->2*x 或 x->2*x //03 接收兩個參數(shù)(數(shù)字),并返回他們的差值。 (int x , int y)-> x - y //接收一個String對象,并在控制臺打印,不返回任何值,有點像void (String s)->System.out.print(s)
Lambda簡化Runnable例子
**原本的實現(xiàn)方式:**可以使用匿名內(nèi)部類的方式創(chuàng)建一個線程,這已經(jīng)是面向?qū)ο蟮木幊谭椒茏龅降淖詈喕臅鴮懥?/p>
new Thread(new Runnable() { @Override public void run() { System.out.println(Thread.currentThread().getName()+"線程啟動成功"); } }).start();
代碼分析
- Thread類需要Runnable接口作為參數(shù),其中的抽象run方法才是用來指定線程任務(wù)的核心。
- 為了指定run方法體,不得不創(chuàng)建Runnable接口的實現(xiàn)類。
- 為了省去定義實現(xiàn)類的麻煩,不得不使用匿名內(nèi)部類。
- 為了覆蓋重寫原先在Runnable中的抽象方法run(),不得不再寫一遍方法名稱,方法參數(shù),方法返回值,而且不能寫錯。
- 但經(jīng)過分析好像只有run()中的方法體才是關(guān)鍵。
Lambda表達式的實現(xiàn):
new Thread(()-> { System.out.println(Thread.currentThread().getName()+"線程啟動成功"); }).start();
我們刪去了前面提到的匿名內(nèi)部類里面冗余的所有代碼,只保留核心的任務(wù)——傳遞一個語句到Thread中執(zhí)行。這樣的寫法在JDK1.8及以后都能編譯通過,功能和前面寫的匿名內(nèi)部類的實現(xiàn)是一樣的。、
自定義接口實現(xiàn)lambda
先定義一個只有一個抽象方法的接口:
public interface Cook { void makeFood(); //抽象方法無參無返回值 }
再在測試類中按格式定義一個靜態(tài)方法并在主函數(shù)中書寫Lambda表達式
public class cookDemo { public static void main(String[] args) { invokeCook(()->System.out.println("做食物ing")); //調(diào)用定義的靜態(tài)方法 } public static void invokeCook(Cook cook){ cook.makeFood(); }//按照這樣的格式書寫靜態(tài)方法 }
函數(shù)式接口
概念
函數(shù)式接口在java中中指,有且只有一個抽象方法的接口。
函數(shù)式接口,即適用于函數(shù)式編程場景的接口。而Java中函數(shù)式編程的體現(xiàn)就是Lambda,所以函數(shù)式接口就是可以適用于Lambda使用的接口。只有確保接口中有且只有一個抽象方法,Java中的lambda才能順利推導(dǎo)。
備注
“語法糖”是指使用更加方便,但原理不變的代碼語法。例如在遍歷集合是使用的for-each語法,其實它的底層的原理實現(xiàn)仍然是迭代器,這便是語法糖,從應(yīng)用層面來說,Java中的Lambda可以被當(dāng)作匿名內(nèi)部類的“語法糖”,但兩者在原理上是不一樣的。 Lambda大致原理可能是,在編譯的過程中動態(tài)生成一個內(nèi)部類和一個靜態(tài)私用方法,然后在使用lambda的地方調(diào)用這個生成的靜態(tài)方法,這個靜態(tài)方法的內(nèi)容就和函數(shù)式接口中函數(shù)要實現(xiàn)的功能一致。
格式
只要確保接口中有且只有一個抽象方法即可:
修飾符 interface 接口名稱 { public abstract 返回值類型 方法名稱 (參數(shù)列表); //其他非抽象方法內(nèi)容,我們知道還可以放默認方法,靜態(tài)方法 }
其中抽象方法的 public abstract 是可以被省略的,所以定義一個函數(shù)式接口很簡單:
public interface MyFunctionInterface { void myMethod(); }
@FunctionalInterface注解
與@Override 注解的作用類似,java8中專門給函數(shù)式接口引進了一個新的注解:@FunctionalInterface。該注解可以用于一個接口的定義上:
@FunctionalInterface public interface MyFunctionalInterface{ void myMethod(); }
一旦使用該注解來定義接口,編譯器就會強制檢查該接口,是否確實有且只有一個抽象方法,否則就會編譯報錯。需要注意的是,即使不使用該注解,只要滿足函數(shù)式接口的定義規(guī)范,這仍然是一個函數(shù)式接口,使用起來都一樣。
自定義函數(shù)式接口
對于剛剛定義好的MyFunctionalInterface函數(shù)式接口,典型的應(yīng)用場景就是作為方法的參數(shù):
public class DemoFunctionalInterface{ //使用自定義的函數(shù)式接口作為方法參數(shù) private static void testMethod(MyFunctionalInterface inter){ inter.myMethod; //調(diào)用自定義的函數(shù)式接口中的方法 } public static void main(String[] args){ //調(diào)用函數(shù)式接口方法 testMethod(()->System.out.println("lambda執(zhí)行成功")); } }
到此這篇關(guān)于Java特性 Lambda 表達式和函數(shù)式接口的文章就介紹到這了,更多相關(guān)Java Lambda 內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot與SpringSecurity整合方法附源碼
這篇文章主要介紹了SpringBoot與SpringSecurity整合,本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-01-01

Java 數(shù)據(jù)庫時間返回前端顯示錯誤(差8個小時)的解決方法

Spring Boot利用JSR303實現(xiàn)參數(shù)驗證的方法實例