函數式接口(functional interface)就是一個有且僅有一個抽象方法,但是可以有多個非抽象方法的接口。
函數式接口可以被隱式轉換為lambda表達式。
函數式接口可以現有的函數友好地支持 lambda。
介紹
函數式接口其實就是一個抽象接口類,在java 8之前已有的函數式接口有以下。
1
2
3
|
java.lang.runnable java.util.concurrent.callable java.util.comparator |
等等...
使用方法
其實上述所說的接口類只需要使用functionalinterface注解修飾,就成為了java中的函數式接口。比如jdk中callable接口定義
1
2
3
4
|
@functionalinterface public interface callable<v> { v call() throws exception; } |
就這么簡單。
現在來講講java 8新增function接口。下面是其定義
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
// t 是傳入參數 // r 是返回參數 @functionalinterface public interface function<t, r> { r apply(t t); default <v> function<v, r> compose(function<? super v, ? extends t> before { objects.requirenonnull(before); return (v v) -> apply(before.apply(v)); } default <v> function<t, v> andthen(function<? super r, ? extends v> after) { objects.requirenonnull(after); return (t t) -> after.apply(apply(t)); } static <t> function<t, t> identity() { return t -> t; } } |
他可以理解為c語言中的函數指針(個人看法)。
在實際使用中,apply方法使用比較廣泛。compose/andthen多用于兩個以上函數接口并且執行上有先后順序的場景下。
在具體業務代碼中,我一般結合bifuncton/supplier一起使用。bifunction支持2個參數的,function只支持一個參數。supplier可以用來存儲具體需要的值,通過get來獲取。
例子
引用平時工作的代碼。該例子主要規避了多個判斷條件if/else造成代碼臃腫的,同時也可以把相同的業務邏輯抽象出函數接口,從而可以在多處代碼重用。具體代碼如下。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
function<object, integer> actiontest1 = (object) -> { // logic return 0 ; }; function<object, integer> actiontest2 = (object) -> { // logic return 0 ; }; public supplier<map<integer, function<object, integer>>> actionsupplier = () -> { map<integer, function<object, integer>> maps = new hashmap<>(); maps.put( 1 , actiontest1); maps.put( 2 , actiontest2); return maps; }; // 具體使用 public void test( int type, object object) { optional.ofnullable(actionsupplier.get().get(type)).ifpresent(x -> x.apply(v, object)); // if/else 邏輯 if (type == 1 ) { // test1 logic } else if (type == 2 ) { // test2 logic } } |
總結
個人認為,在業務邏輯分支判斷較多的場景,是比較適合使用function的,而且還有以下幾點好處
- 代碼看上去比較簡潔
- 相同邏輯可以重用(當然封裝成函數也是可以的~)
- 代碼后期比較好維護點。
函數式接口實例
predicate <t> 接口是一個函數式接口,它接受一個輸入參數 t,返回一個布爾值結果。
該接口包含多種默認方法來將predicate組合成其他復雜的邏輯(比如:與,或,非)。
該接口用于測試對象是 true 或 false。
我們可以通過以下實例(java8tester.java)來了解函數式接口 predicate <t> 的使用:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
|
java8tester.java 文件 import java.util.arrays; import java.util.list; import java.util.function.predicate; public class java8tester { public static void main(string args[]){ list<integer> list = arrays.aslist( 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 ); // predicate<integer> predicate = n -> true // n 是一個參數傳遞到 predicate 接口的 test 方法 // n 如果存在則 test 方法返回 true system.out.println( "輸出所有數據:" ); // 傳遞參數 n eval(list, n-> true ); // predicate<integer> predicate1 = n -> n%2 == 0 // n 是一個參數傳遞到 predicate 接口的 test 方法 // 如果 n%2 為 0 test 方法返回 true system.out.println( "輸出所有偶數:" ); eval(list, n-> n% 2 == 0 ); // predicate<integer> predicate2 = n -> n > 3 // n 是一個參數傳遞到 predicate 接口的 test 方法 // 如果 n 大于 3 test 方法返回 true system.out.println( "輸出大于 3 的所有數字:" ); eval(list, n-> n > 3 ); } public static void eval(list<integer> list, predicate<integer> predicate) { for (integer n: list) { if (predicate.test(n)) { system.out.println(n + " " ); } } } } |
執行以上腳本,輸出結果為:
$ javac java8tester.java
$ java java8tester
輸出所有數據:
1
2
3
4
5
6
7
8
9
輸出所有偶數:
2
4
6
8
輸出大于 3 的所有數字:
4
5
6
7
8
9
總結
以上所述是小編給大家介紹的java 8 function函數式接口及函數式接口實例,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對服務器之家網站的支持!
原文鏈接:https://www.jianshu.com/p/bbe30b42135f