以下我們繼續(xù)深入Java8函數(shù)式編程模型
1
2
3
4
5
6
7
8
9
10
11
|
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); } }); } } |
這段程序很簡單,首先初始化一個Integer類型的集合然后向控制臺輸出每個元素。其中我們注意到forEach方法,它就是Java8中新增加的默認方法。
1
2
3
4
5
6
7
8
9
10
11
|
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修飾。這樣任何一個該接口的子類型都可以繼承forEach方法的實現(xiàn),所以List接口因為是Iterable的間接子接口,所以也繼承了該默認方法。Java8采用這種巧妙的方式既擴展了接口的功能,又兼容了老版本。
接下來分析下forEach的實現(xiàn),首先接收了一個Consumer類型的參數(shù)action,進行非空判斷,然后遍歷當(dāng)前所有元素交由action的accept方法進行處理。那么Consumer又是什么鬼,看源碼
1
2
3
4
5
6
7
8
9
10
11
12
13
|
@FunctionalInterface public interface Consumer< T > { /** * Performs this operation on the given argument. * * @param t the input argument */ void accept(T t); . .省略 . } |
一個接口,有且僅有一個抽象方法,被@FunctionalInterface修飾,典型的函數(shù)式接口。
ok,現(xiàn)在我們知道forEach接收的Consumer類型的參數(shù)是一個函數(shù)式接口,接口里唯一的accept抽象方法接收一個參數(shù),不返回值。那通過上一篇文章我們知道,創(chuàng)建函數(shù)式接口類型的實例其中一種方式是使用Lambda表達式,所以可以將最上面的程序改造一下
1
2
3
4
5
6
7
|
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表達式 接收一個參數(shù) 不返回值 list.forEach(item -> System.out.println(item)); } } |
該lambda表達式item -> System.out.println(item)接收一個參數(shù) 不返回值,符合accept方法簽名要求,編譯通過。
也就是說如果使用lambda表達式來創(chuàng)建一個函數(shù)式接口實例,那這個lambda表達式的入?yún)⒑头祷乇仨毞线@個函數(shù)式接口中唯一的抽象方法的方法簽名。
接下來再對程序進行改造
1
2
3
4
5
6
7
|
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后面有兩個冒號,反正當(dāng)時我是凌亂了。。。這個就是函數(shù)式接口實例第二種創(chuàng)建方式:方法引用方法引用的語法是 對象::方法名
同樣,使用方法引用方式去創(chuàng)建函數(shù)式接口實例也必須遵守方法簽名的定義,看下此處println方法源碼
1
2
3
4
5
6
7
|
public void println(Object x) { String s = String.valueOf(x); synchronized (this) { print(s); newLine(); } } |
接收一個參數(shù),并不返回值,編譯通過。
最后我們來看下創(chuàng)建函數(shù)式接口的最后一種,第三種方式:構(gòu)造方法引用 ,繼續(xù)改程序
1
2
3
4
5
6
7
8
9
10
11
|
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òu)造方法,該構(gòu)造方法接收一個參數(shù),不返回值,編譯通過。(僅為展示構(gòu)造方法引用的用法)
結(jié)合上一篇文章可以總結(jié)一下,創(chuàng)建函數(shù)式接口類型的三種方式:
1.lambda表達式
2.方法引用
3.構(gòu)造方法引用
注意:無論是哪種方式,必須要符合抽象方法的方法簽名