實(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); } }); } }
這段程序很簡單,首先初始化一個(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ò)展了接口的功能,又兼容了老版本。
接下來分析下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á)式來創(chuàng)建一個(gè)函數(shù)式接口實(shí)例,那這個(gè)lambda表達(dá)式的入?yún)⒑头祷乇仨毞线@個(gè)函數(shù)式接口中唯一的抽象方法的方法簽名。
接下來再對(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)建方式:方法引用方法引用的語法是 對(duì)象::方法名
同樣,使用方法引用方式去創(chuàng)建函數(shù)式接口實(shí)例也必須遵守方法簽名的定義,看下此處println方法源碼
public void println(Object x) { String s = String.valueOf(x); synchronized (this) { print(s); newLine(); } }
接收一個(gè)參數(shù),并不返回值,編譯通過。
最后我們來看下創(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)造方法引用的語法是:類名::new
我們給Test1新添加了一個(gè)構(gòu)造方法,該構(gòu)造方法接收一個(gè)參數(shù),不返回值,編譯通過。(僅為展示構(gòu)造方法引用的用法)
結(jié)合上一篇文章可以總結(jié)一下,創(chuàng)建函數(shù)式接口類型的三種方式:
1.lambda表達(dá)式
2.方法引用
3.構(gòu)造方法引用
注意:無論是哪種方式,必須要符合抽象方法的方法簽名
相關(guān)文章
Java中MyBatis的動(dòng)態(tài)語句詳解
這篇文章主要介紹了Java中MyBatis的動(dòng)態(tài)語句詳解,動(dòng)態(tài) SQL 是 MyBatis 的強(qiáng)大特性之一,通過不同參數(shù)生成不同的 SQL,可以動(dòng)態(tài)地對(duì)數(shù)據(jù)持久層進(jìn)行操作,而不需要每個(gè)數(shù)據(jù)訪問操作都要進(jìn)行手動(dòng)地拼接 SQL 語句,需要的朋友可以參考下2023-08-08解決@Api注解不展示controller內(nèi)容的問題
這篇文章主要介紹了解決@Api注解不展示controller內(nèi)容的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。2022-01-01springboot如何讀取自定義properties并注入到bean中
這篇文章主要介紹了springboot讀取自定義properties并注入到bean中,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11Springboot集成spring data elasticsearch過程詳解
這篇文章主要介紹了springboot集成spring data elasticsearch過程詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-04-04