Java的函數(shù)式編程詳解
前言
雖然Jdk版本已經(jīng)出到了20+,甚至IDEA創(chuàng)建maven默認(rèn)項(xiàng)目Jdk版本都11了,但是你發(fā)任你發(fā),我用Java8,最近別的組的同事要對接個(gè)需求,把代碼倉庫開給他之后反饋看不太明白代碼。
用了這么久的Java8,我尋思這種話也好意思說出來嗎,難道自己是PythonBoy出身就是看不懂Java的理由嗎,身為一個(gè)合格的后端Boy不會還有人看不明白Java的函數(shù)式編程吧。
函數(shù)式編程
那什么是函數(shù)編程呢,說白了就是可以把函數(shù)作為參數(shù)傳遞給其他函數(shù),亦或者將一個(gè)函數(shù)作為返回值從另一個(gè)函數(shù)中返回。
于是Jdk根據(jù)不同的場景提供了一些核心特性和函數(shù)式接口:
Consumer<T>:接受一個(gè)輸入?yún)?shù) T,并在方法內(nèi)部執(zhí)行操作,沒有返回值。 Supplier<T>:不接受任何輸入?yún)?shù),提供一個(gè)結(jié)果 T。 Function<T, R>:接受一個(gè)輸入?yún)?shù) T,執(zhí)行操作并返回一個(gè)結(jié)果 R。 Predicate<T>:接受一個(gè)輸入?yún)?shù) T,返回一個(gè)布爾值表示是否滿足條件。 UnaryOperator<T>:接受一個(gè)輸入?yún)?shù) T,執(zhí)行操作并返回一個(gè)與輸入?yún)?shù)類型相同的結(jié)果。 BinaryOperator<T>:接受兩個(gè)相同類型的輸入?yún)?shù) T,執(zhí)行操作并返回一個(gè)相同類型的結(jié)果。
相信大家經(jīng)常在業(yè)務(wù)代碼中有過這樣的場景,DAO層返回了一個(gè)Optional類,如果我們只想獲得其中的某一個(gè)屬性應(yīng)該怎么寫。
我們借助一個(gè)小菜雞老弟的代碼說,這段代碼寫的只能說一言難盡,按照他的想法,他只想拿到age字段,其實(shí)我們只需要借助map即可滿足需求。
aa.map(Studnet::getAge) .orElseThrow( () -> throw new Exception("user not found") );
這里的orElseThrow其實(shí)就是傳進(jìn)去的一個(gè)方法,我們可以在方法里做些別的事,比如打印下信息之類的
aa.map(Student::getAge).orElseThrow(() -> { System.out.println("error! error!"); return new Exception("user not found"); });
這就是一個(gè)很簡單的函數(shù)式編程例子,把函數(shù)作為參數(shù),在另一個(gè)函數(shù)中調(diào)用。這里舉一個(gè)真實(shí)的業(yè)務(wù)案例,根據(jù)高內(nèi)聚低耦合的抽象思想,我們保存數(shù)據(jù)的時(shí)候做一層前置數(shù)據(jù)校驗(yàn),不同的類有不同的檢查字段和方法,比如Student類要檢查年齡是否大于18歲。
public class Java8Test { public static void main(String[] args) { Student student1 = Student.builder().age(10).name("uptown").build(); savePeople(student1, (Student s1) -> { if (s1.getAge() < 18) { throw new RuntimeException("18歲禁!"); } }); } public static <T extends People> void savePeople(T t, Consumer<T> function) { function.accept(t); // 業(yè)務(wù)TODO } } @Data @Builder class People { } @Data @Builder class Student extends People { Integer age; String name; }
這樣寫是不是比疊加if語句優(yōu)化很多。
閉包
熟練使用函數(shù)式編程的jym對這個(gè)概念肯定不陌生,閉包就是能夠讀取其他函數(shù)內(nèi)部變量的函數(shù),借助網(wǎng)上js閉包的例子。
function makeFunc() { var name = "Mozilla"; function displayName() { console.log(name); } return displayName; } var myFunc = makeFunc(); myFunc(); // "Mozilla"
可以通俗的想象成,你在房間里寫了一段代碼包含一個(gè)函數(shù)和一些變量。當(dāng)你離開房間后,函數(shù)和變量仍然存在,它們就形成了一個(gè)閉包。當(dāng)你再次進(jìn)入房間時(shí),你可以使用閉包中的函數(shù)來訪問和修改之前已經(jīng)修改過的變量,變量仍然記錄著之前的狀態(tài)。
那么Java中是否有閉包特性呢,答案是嚴(yán)格意義上說沒有。
因?yàn)樽屑?xì)想一下就發(fā)現(xiàn)在JVM里這套邏輯顯然不符合,在棧中的變量函數(shù)結(jié)束后就會被清掉,怎么會記住之前的結(jié)果。但我們可以通過匿名內(nèi)部類或lambad表達(dá)式實(shí)現(xiàn),lambad本質(zhì)上也是匿名內(nèi)部類。
public static void main(String[] args) { final int i = 0; Supplier<Integer> sup = new Supplier<>(){ Integer get(){ return i; } }; }
為了保證正確性和一致性,JDK在語法上規(guī)定了lambad內(nèi)部訪問的局部變量必須是final
修飾的。因?yàn)?strong>lambda表達(dá)式本質(zhì)上創(chuàng)建了一個(gè)閉包,捕獲了外部的局部變量,并在Lambda表達(dá)式的函數(shù)體中使用這些變量。
這只是其中一點(diǎn),變量修飾為final還有一些數(shù)據(jù)一致方面的問題,這里就不再贅述了。
以上就是Java的函數(shù)式編程詳解的詳細(xì)內(nèi)容,更多關(guān)于Java函數(shù)式編程的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
java實(shí)現(xiàn)學(xué)生信息錄入界面
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)學(xué)生信息錄入界面,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-04-04java的三種隨機(jī)數(shù)生成方式的實(shí)現(xiàn)方法
這篇文章主要介紹了java的三種隨機(jī)數(shù)生成方式的實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-09-09spring-boot中spring-boot-maven-plugin報(bào)紅錯誤及解決
這篇文章主要介紹了spring-boot中spring-boot-maven-plugin報(bào)紅錯誤及解決方案,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-03-03MyBatis-Plus 插件擴(kuò)展的實(shí)現(xiàn)
MyBatis-Plus通過插件擴(kuò)展機(jī)制增強(qiáng)功能,基于MyBatis Interceptor攔截器,包括分頁插件、邏輯刪除、SQL性能分析和樂觀鎖等,開發(fā)者可自定義插件以適應(yīng)特定需求,有效地增強(qiáng)SQL執(zhí)行過程的控制和優(yōu)化,同時(shí)注意插件使用的性能影響和執(zhí)行順序2024-09-09