實(shí)例詳解Java8函數(shù)式接口
以下我們繼續(xù)深入Java8函數(shù)式編程模型
public class Test1 {
public static void main(String[] args) {
List<Integer> list = Arrays.asList(1,2,3,4,5,6,7,8,9,10);
list.forEach(new Consumer<Integer>() {
@Override
public void accept(Integer integer) {
System.out.println(integer);
}
});
}
}
這段程序很簡(jiǎn)單,首先初始化一個(gè)Integer類型的集合然后向控制臺(tái)輸出每個(gè)元素。其中我們注意到forEach方法,它就是Java8中新增加的默認(rèn)方法。
public interface Iterable<T> {
.
.省略
.
default void forEach(Consumer<? super T> action) {
Objects.requireNonNull(action);
for (T t : this) {
action.accept(t);
}
}
}
它被聲明在Iterable接口中,并被關(guān)鍵字default修飾。這樣任何一個(gè)該接口的子類型都可以繼承forEach方法的實(shí)現(xiàn),所以List接口因?yàn)槭荌terable的間接子接口,所以也繼承了該默認(rèn)方法。Java8采用這種巧妙的方式既擴(kuò)展了接口的功能,又兼容了老版本。
接下來(lái)分析下forEach的實(shí)現(xiàn),首先接收了一個(gè)Consumer類型的參數(shù)action,進(jìn)行非空判斷,然后遍歷當(dāng)前所有元素交由action的accept方法進(jìn)行處理。那么Consumer又是什么鬼,看源碼
@FunctionalInterface
public interface Consumer<T> {
/**
* Performs this operation on the given argument.
*
* @param t the input argument
*/
void accept(T t);
.
.省略
.
}
一個(gè)接口,有且僅有一個(gè)抽象方法,被@FunctionalInterface修飾,典型的函數(shù)式接口。
ok,現(xiàn)在我們知道forEach接收的Consumer類型的參數(shù)是一個(gè)函數(shù)式接口,接口里唯一的accept抽象方法接收一個(gè)參數(shù),不返回值。那通過上一篇文章我們知道,創(chuàng)建函數(shù)式接口類型的實(shí)例其中一種方式是使用Lambda表達(dá)式,所以可以將最上面的程序改造一下
public class Test1 {
public static void main(String[] args) {
List<Integer> list = Arrays.asList(1,2,3,4,5,6,7,8,9,10);
//Lambda表達(dá)式 接收一個(gè)參數(shù) 不返回值
list.forEach(item -> System.out.println(item));
}
}
該lambda表達(dá)式item -> System.out.println(item)接收一個(gè)參數(shù) 不返回值,符合accept方法簽名要求,編譯通過。
也就是說如果使用lambda表達(dá)式來(lái)創(chuàng)建一個(gè)函數(shù)式接口實(shí)例,那這個(gè)lambda表達(dá)式的入?yún)⒑头祷乇仨毞线@個(gè)函數(shù)式接口中唯一的抽象方法的方法簽名。
接下來(lái)再對(duì)程序進(jìn)行改造
public class Test1 {
public static void main(String[] args) {
List<Integer> list = Arrays.asList(1,2,3,4,5,6,7,8,9,10);
//方法引用
list.forEach(System.out::println);
}
}
看到out后面有兩個(gè)冒號(hào),反正當(dāng)時(shí)我是凌亂了。。。這個(gè)就是函數(shù)式接口實(shí)例第二種創(chuàng)建方式:方法引用方法引用的語(yǔ)法是 對(duì)象::方法名
同樣,使用方法引用方式去創(chuàng)建函數(shù)式接口實(shí)例也必須遵守方法簽名的定義,看下此處println方法源碼
public void println(Object x) {
String s = String.valueOf(x);
synchronized (this) {
print(s);
newLine();
}
}
接收一個(gè)參數(shù),并不返回值,編譯通過。
最后我們來(lái)看下創(chuàng)建函數(shù)式接口的最后一種,第三種方式:構(gòu)造方法引用 ,繼續(xù)改程序
public class Test1 {
public static void main(String[] args) {
List<Integer> list = Arrays.asList(1,2,3,4,5,6,7,8,9,10);
//構(gòu)造方法引用
list.forEach(Test1::new);
}
Test1(Integer i){
System.out.println(i);
}
}
構(gòu)造方法引用的語(yǔ)法是:類名::new
我們給Test1新添加了一個(gè)構(gòu)造方法,該構(gòu)造方法接收一個(gè)參數(shù),不返回值,編譯通過。(僅為展示構(gòu)造方法引用的用法)
結(jié)合上一篇文章可以總結(jié)一下,創(chuàng)建函數(shù)式接口類型的三種方式:
1.lambda表達(dá)式
2.方法引用
3.構(gòu)造方法引用
注意:無(wú)論是哪種方式,必須要符合抽象方法的方法簽名
相關(guān)文章
Java基本類型與byte數(shù)組之間相互轉(zhuǎn)換方法
下面小編就為大家?guī)?lái)一篇Java基本類型與byte數(shù)組之間相互轉(zhuǎn)換方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來(lái)看看吧2016-08-08
maven中resource配置的實(shí)現(xiàn)示例
我們?cè)谑褂肕aven組件來(lái)構(gòu)建項(xiàng)目的時(shí)候,通常將配置文件放在資源文件目錄下,本文主要介紹了maven中resource配置的實(shí)現(xiàn)示例,具有一定的參考價(jià)值,感興趣的可以了解一下2023-09-09
Jmeter如何獲取jtl文件中所有的請(qǐng)求報(bào)文詳解
JMeter的可以創(chuàng)建一個(gè)包含測(cè)試運(yùn)行結(jié)果的文本文件,這些通常稱為JTL文件,因?yàn)檫@是默認(rèn)擴(kuò)展名,但可以使用任何擴(kuò)展名,這篇文章主要給大家介紹了關(guān)于Jmeter如何獲取jtl文件中所有的請(qǐng)求報(bào)文的相關(guān)資料,需要的朋友可以參考下2021-09-09
IntelliJ IDEA安裝scala插件并創(chuàng)建scala工程的步驟詳細(xì)教程
這篇文章主要介紹了IntelliJ IDEA安裝scala插件并創(chuàng)建scala工程的步驟,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-07-07
SpringBoot多環(huán)境切換的靈活配置詳細(xì)教程
在真實(shí)項(xiàng)目開發(fā)的時(shí)候,一定會(huì)有多個(gè)環(huán)境,下面這篇文章主要給大家介紹了關(guān)于SpringBoot多環(huán)境切換靈活配置的相關(guān)資料,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-04-04
JVM內(nèi)存結(jié)構(gòu)劃分實(shí)例解析
這篇文章主要介紹了JVM內(nèi)存結(jié)構(gòu)劃分實(shí)例解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12

