4個(gè)Java8中你需要知道的函數(shù)式接口分享
前言
Java 8 中提供了許多函數(shù)式接口,包括Function、Consumer、Supplier、Predicate 等等。這 4 個(gè)接口就是本篇將要分享的內(nèi)容,它們都位于 java.util.function
包下。
為什么需要知道這幾個(gè)函數(shù)式接口
因?yàn)檫@ 4 個(gè)函數(shù)式接口是 Java 8 中新增的重要接口,同時(shí) Java 8 的 Stream 新特性,也有用到這些接口,所以學(xué)習(xí)它們可以幫助我們更好地理解 Stream 流。
也正因?yàn)檫@是函數(shù)式接口,所以就可以使用 Lambda 表達(dá)式來寫接口的實(shí)現(xiàn)邏輯。而且學(xué)習(xí)的過程中可以更好地理解函數(shù)式編程的思想。
Function 接口
說明
Function 這個(gè)單詞的意思就有「函數(shù)」的意思,就數(shù)學(xué)中的 y = f(x),接收一個(gè) x 參數(shù),通過函數(shù) f 運(yùn)算后,返回一個(gè)結(jié)果 y。
Function
接口包含四個(gè)方法:
apply(T t)
:這是Function
接口的主要方法,它接收一個(gè)參數(shù)并返回一個(gè)結(jié)果。同時(shí)它也是唯一的抽象的方法,剩下的都是有默認(rèn)實(shí)現(xiàn)的(Java 8 中接口的抽象方法支持默認(rèn)實(shí)現(xiàn))。andThen(Function after)
:作用是將兩個(gè)Function
組合。首先執(zhí)行當(dāng)前函數(shù),再執(zhí)行andThen
函數(shù),并將當(dāng)前函數(shù)的結(jié)果作為參數(shù)傳遞給andThen
函數(shù)。compose(Function before)
:同理,將兩個(gè)Function
組合,將先執(zhí)行compose
函數(shù),再執(zhí)行當(dāng)前函數(shù),并將compose
函數(shù)的結(jié)果作為參數(shù)傳遞給當(dāng)前函數(shù)。identity()
: 返回一個(gè)執(zhí)行恒等轉(zhuǎn)換的函數(shù),即返回輸入?yún)?shù)本身。
Function 接口通常用于將一個(gè)類型的值轉(zhuǎn)換為另一個(gè)類型的值。
apply 方法
// Function 接口的泛型,第一個(gè)參數(shù)是入?yún)㈩愋?,第二個(gè)參數(shù)是出參類型 // Function 接口只有一個(gè)抽象方法,就是 apply(),下面利用 Lambda 表達(dá)式實(shí)現(xiàn)這個(gè)抽象方法并創(chuàng)建 Function 對象 Function<Integer, String> function = num -> "GTA" + num; // 將5這個(gè)參數(shù)傳遞給function,得到返回結(jié)果 String result = function.apply(5); System.out.println(result); // 打?。篏TA5
andThen 和 compose 方法
// 定義兩個(gè) Function 對象進(jìn)行相關(guān)轉(zhuǎn)換操作 Function<String, String> upperCase = s -> s.toUpperCase(); Function<String, String> addPostfix = s -> s + "5"; // 鏈?zhǔn)秸{(diào)用,將 gta 這個(gè)字符串參數(shù)先傳遞 upperCase 這個(gè)函數(shù)進(jìn)行操作,然后將得到的結(jié)果傳遞給 addPostfix 函數(shù)進(jìn)行操作,得到返回結(jié)果 String str = upperCase.andThen(addPostfix).apply("gta"); System.out.println(str); // 打?。篏TA5
identify 方法
identity
方法返回一個(gè)執(zhí)行恒等轉(zhuǎn)換的函數(shù),該函數(shù)將輸入?yún)?shù)原樣返回。例如:
Function<String, String> identity = Function.identity(); String result = identity.apply("hello"); // result is "hello"
Consumer 接口
說明
Consumer 這個(gè)單詞的意思就有「消費(fèi)者」的意思,就把入?yún)⑾M(fèi)了,并不會(huì)返回結(jié)果給你。
Consumer 接口包含兩個(gè)方法:
accept(T t)
:該方法接受一個(gè)參數(shù)并執(zhí)行一些操作。andThen(Consumer after)
:同理,將兩個(gè) Consumer 組合,先后進(jìn)行消費(fèi)。
accept 方法
Consumer 接口通常用于消費(fèi)一個(gè)參數(shù)然后執(zhí)行一些操作。例如:
// Consumer 接口,泛型參數(shù)是入?yún)㈩愋停邮芤粋€(gè)參數(shù),并不返回結(jié)果,相當(dāng)于消費(fèi)了這個(gè)參數(shù) Consumer<String> consumer = s -> System.out.println(s); consumer.accept("我輸入什么就打印什么"); // 打?。何逸斎胧裁淳痛蛴∈裁?
andThen 方法
組合兩個(gè) Consumer:
Consumer<String> first = s -> System.out.println(s + 5); Consumer<String> second = s -> System.out.println(s + 6); // 先執(zhí)行 first 這個(gè) Consumer,接著執(zhí)行 second 這個(gè) Consumer Consumer<String> combination = first.andThen(second); combination.accept("GTA"); // 打印:GTA5 GTA6
Supplier 接口
Supplier 接口只定義了一個(gè) get()
方法,該方法不接受任何參數(shù)并返回一個(gè)結(jié)果。
Supplier 這個(gè)單詞的意思就有「供應(yīng)者」的意思,給我的感覺就是生產(chǎn)者,不用參數(shù),直接生產(chǎn)一個(gè)東西給你。
Supplier 接口通常用于生成一個(gè)值。例如:
// Supplier 接口,泛型參數(shù)是出參類型,不接受參數(shù),但是會(huì)提供結(jié)果,相當(dāng)于生產(chǎn)了某個(gè)東西 Supplier<String> supplier = () -> "提供一個(gè)我隨便打的字符串給調(diào)用方"; String text = supplier.get(); System.out.println(text); // 打印:提供一個(gè)我隨便打的字符串給調(diào)用方
Predicate 接口
說明
Predicate 這個(gè)單詞的意思就有「預(yù)言,預(yù)測,謂語,謂詞」的意思,就是用來預(yù)測判斷的。
Predicate
接口包含四個(gè)方法:
test(T t)
:該方法接受一個(gè)參數(shù)并返回一個(gè)布爾值。and(Predicate other)
:與另一個(gè) Predicate 進(jìn)行組合,實(shí)現(xiàn)邏輯與操作。negate()
:與另一個(gè) Predicate 進(jìn)行組合,實(shí)現(xiàn)邏輯非操作。or(Predicate other)
:與另一個(gè) Predicate 進(jìn)行組合,實(shí)現(xiàn)邏輯或操作。
test方法
Predicate 接口通常用于測試一個(gè)條件是否成立。例如:
// Predicate 接口,泛型參數(shù)是入?yún)㈩愋?,返回布爾? Predicate<String> predicate = s -> s.contains("god23bin"); boolean flag = predicate.test("god23bin能給你帶來收獲嗎?"); System.out.println("god23bin能給你帶來收獲嗎?" + flag); // 打?。篻od23bin能給你帶來收獲嗎?true
and 方法
為了便于演示,這里準(zhǔn)備兩個(gè) Predicate:
Predicate<String> startsWithA = (str) -> str.startsWith("A"); // 如果傳入的字符串是A開頭,則返回 true Predicate<String> endsWithZ = (str) -> str.endsWith("Z"); // 如果傳入的字符串是Z結(jié)尾,則返回 true
使用 and 進(jìn)行組合,與操作:
Predicate<String> startsWithAAndEndsWithZ = startsWithA.and(endsWithZ); System.out.println(startsWithAAndEndsWithZ.test("ABCDEFZ")); // true System.out.println(startsWithAAndEndsWithZ.test("BCDEFGH")); // false
negate 方法
使用 negate 進(jìn)行組合,非操作:
Predicate<String> notStartsWithA = startsWithA.negate(); System.out.println(notStartsWithA.test("ABCDEF")); // false System.out.println(notStartsWithA.test("BCDEFGH")); // true
or 方法
使用 or 進(jìn)行組合,或操作:
Predicate<String> startsWithAOrEndsWithZ = startsWithA.or(endsWithZ); System.out.println(startsWithAOrEndsWithZ.test("ABCDEF")); // true System.out.println(startsWithAOrEndsWithZ.test("BCDEFGH")); // false
那這些接口有什么應(yīng)用呢
在 Stream 流中就有應(yīng)用上這些函數(shù)式接口。當(dāng)然,當(dāng)你有相似的需求時(shí),你自己也可以應(yīng)用上這些接口。下面說下 Stream 流中的應(yīng)用。
Function 接口:例如 map 方法,map 方法就是將一個(gè)類型的值轉(zhuǎn)換為另一個(gè)類型的值。
// map 方法,將 T 類型的值轉(zhuǎn)換成 R 類型的值 // R 是返回的 Stream 流的元素類型,T 是原先 Stream 流的元素類型 <R> Stream<R> map(Function<? super T, ? extends R> mapper);
Consumer 接口:例如 forEach 方法
// forEach 方法,遍歷 Stream 流中的元素,T 類型是 Stream 流的元素類型 void forEach(Consumer<? super T> action);
Supplier 接口:例如 generate 方法
// 生成一個(gè)無限長度的 Stream 流 public static<T> Stream<T> generate(Supplier<T> s) { Objects.requireNonNull(s); return StreamSupport.stream( new StreamSpliterators.InfiniteSupplyingSpliterator.OfRef<>(Long.MAX_VALUE, s), false); }
Predicate 接口:例如 filter 方法,使用 Predicate 進(jìn)行過濾操作。
// 過濾出 Stream 流中,判斷結(jié)果為 true 的元素 Stream<T> filter(Predicate<? super T> predicate);
到此這篇關(guān)于4個(gè)Java8中你需要知道的函數(shù)式接口分享的文章就介紹到這了,更多相關(guān)Java8函數(shù)式接口內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java如何優(yōu)雅的實(shí)現(xiàn)微信登錄注冊
這篇文章主要給大家介紹了關(guān)于Java如何優(yōu)雅的實(shí)現(xiàn)微信登錄注冊的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用java具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2022-02-02詳解SpringBoot開發(fā)使用@ImportResource注解影響攔截器
這篇文章主要介紹了詳解SpringBoot開發(fā)使用@ImportResource注解影響攔截器,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-11-11Java concurrency之集合_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
Java集合主體內(nèi)容包括Collection集合和Map類;而Collection集合又可以劃分為List(隊(duì)列)和Set(集合),有需要的小伙伴可以參考下2017-06-06SpringCloud-Gateway網(wǎng)關(guān)的使用實(shí)例教程
Gateway網(wǎng)關(guān)在微服務(wù)架構(gòu)中扮演了不可或缺的角色,通過集中化管理、智能路由和強(qiáng)大的過濾器機(jī)制,為構(gòu)建高效、可擴(kuò)展的微服務(wù)系統(tǒng)提供了有力支持,這篇文章主要介紹了SpringCloud-Gateway網(wǎng)關(guān)的使用,需要的朋友可以參考下2024-03-03SpringBoot整合rockerMQ消息隊(duì)列詳解
今天和大家一起深入生產(chǎn)級(jí)別消息中間件 - RocketMQ 的內(nèi)核實(shí)現(xiàn),來看看真正落地能支撐萬億級(jí)消息容量、低延遲的消息隊(duì)列到底是如何設(shè)計(jì)的。我會(huì)先介紹整體的架構(gòu)設(shè)計(jì),然后再深入各核心模塊的詳細(xì)設(shè)計(jì)、核心流程的剖析2022-07-07Mockito mock Kotlin Object類方法報(bào)錯(cuò)解決方法
這篇文章主要介紹了Mockito mock Kotlin Object類方法報(bào)錯(cuò)解決方法,本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-09-09windows系統(tǒng)配置Java開發(fā)環(huán)境變量
這篇文章主要介紹了windows系統(tǒng)配置Java開發(fā)環(huán)境變量,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2021-12-12在Spring Boot2中使用CompletableFuture的方法教程
這篇文章主要給大家介紹了關(guān)于在Spring Boot2中使用CompletableFuture的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起看看吧2019-01-01