Java 8 新增函式式介面到底是什麼?
Java 8 新增函式式介面到底是什麼?
從 Java 8 開始便出現了函式式介面(Functional Interface,以下簡稱FI)
定義為: 如果一個介面只有唯一的一個抽象介面,則稱之為函式式介面。為了保證介面符合 FI ,通常會在介面類上新增 @FunctionalInterface 註解。理解了函式式介面可以為 Java 函數語言程式設計打下基礎,最終可通過運用函數語言程式設計極大地提高程式設計效率。
函式式介面 (Functional Interface) 就是一個有且僅有一個抽象方法,但是可以有多個非抽象方法的介面。
函式式介面可以對現有的函式友好地支援 lambda。
JDK 1.8 之前已有的函式式介面:
java.lang.Runnable
java.util.concurrent.Callable
java.security.PrivilegedAction
java.util.Comparator
java.io.FileFilter
java.nio.file.PathMatcher
java.lang.reflect.InvocationHandler
java.beans.PropertyChangeListener
java.awt.event.ActionListener
javax.swing.event.ChangeListener
JDK 1.8 新增加的函式介面:
java.util.function
網上很多教程說新增 4 個函式介面是不對的,java.util.function 它包含了很多類,用來支援 Java的 函數語言程式設計,該包中的函式式介面 43 個,但是最主要的是這四個:
(1)功能性介面:Function<T,R>
(2)斷言性介面:Predicate<T>
(3)供給性介面:Supplier<T>
(4)消費性介面:Consumer<T>
詳細一點介紹:
函式式介面 | 引數型別 | 返回型別 | 用途 |
---|---|---|---|
Consumer | T | void | 對型別T引數操作,無返回結果,包含方法 void accept(T t) |
Supplier | 無 | T | 返回T型別引數,方法時 T get() |
Function | T | R | 對型別T引數操作,返回R型別引數,包含方法 R apply(T t) |
Predicate | T | boolean | 斷言型介面,對型別T進行條件篩選操作,返回boolean,包含方法 boolean test(T t) |
具體的使用:
/**
* Java8內建的四大核心函式式介面:
* Consumer<T>:消費型介面</T>
* Supplier<T>供給型介面</T>
* Function<T,R>函式型介面</T,R>
* Predicate<T>段言型介面</T>
* boolean test(T t)
*/
public class TestLamda3 {
//Consumer<T>
@Test
public void test1(){
happy(10000,(m)-> System.out.println("這次消費了"+m+"元"));
}
public void happy(double money, Consumer<Double> con){
con.accept(money);
}
//Supplier<T>
@Test
public void test2(){
List<Integer> list= getNumList(5,()->{
return (int)Math.random()*100;
});
list.forEach(System.out::println);
}
public List<Integer> getNumList(int num, Supplier<Integer> supplier){
List<Integer> list=new ArrayList<>();
for (int i=0; i<num;i++){
Integer n=supplier.get();
list.add(n);
}
return list;
}
//函式式介面
@Test
public void test4(){
String newStr=strHandle("\t\t\t woshi nide ",(str)->str.trim());
System.out.println(newStr);
}
public String strHandle(String str,Function<String,String> fun){
return fun.apply(str);
}
//段言型介面;將滿足條件的字串放入集合中
@Test
public void test5(){
List<String> list1= Arrays.asList("nihao","hiehei","woai","ni");
List<String> list=filterStr(list1,(s)->s.length()>3);
for (String s : list) {
System.out.println(s);
}
}
public List<String> filterStr(List<String> list, Predicate<String> pre){
List<String> strings=new ArrayList<>();
for (String string : list) {
if(pre.test(string)){
strings.add(string);
}
}
return strings;
}
}
全部介面:
序號 | 介面 & 描述 |
---|---|
1 | BiConsumer<T,U> 代表了一個接受兩個輸入引數的操作,並且不返回任何結果 |
2 | BiFunction<T,U,R> 代表了一個接受兩個輸入引數的方法,並且返回一個結果 |
3 | BinaryOperator<T> 代表了一個作用於於兩個同類型操作符的操作,並且返回了操作符同類型的結果 |
4 | BiPredicate<T,U> 代表了一個兩個引數的boolean值方法 |
5 | BooleanSupplier 代表了boolean值結果的提供方 |
6 | Consumer<T> 代表了接受一個輸入引數並且無返回的操作 |
7 | DoubleBinaryOperator 代表了作用於兩個double值操作符的操作,並且返回了一個double值的結果。 |
8 | DoubleConsumer 代表一個接受double值引數的操作,並且不返回結果。 |
9 | DoubleFunction<R> 代表接受一個double值引數的方法,並且返回結果 |
10 | DoublePredicate 代表一個擁有double值引數的boolean值方法 |
11 | DoubleSupplier 代表一個double值結構的提供方 |
12 | DoubleToIntFunction 接受一個double型別輸入,返回一個int型別結果。 |
13 | DoubleToLongFunction 接受一個double型別輸入,返回一個long型別結果 |
14 | DoubleUnaryOperator 接受一個引數同為型別double,返回值型別也為double 。 |
15 | Function<T,R> 接受一個輸入引數,返回一個結果。 |
16 | IntBinaryOperator 接受兩個引數同為型別int,返回值型別也為int 。 |
17 | IntConsumer 接受一個int型別的輸入引數,無返回值 。 |
18 | IntFunction<R> 接受一個int型別輸入引數,返回一個結果 。 |
19 | IntPredicate 接受一個int輸入引數,返回一個布林值的結果。 |
20 | IntSupplier 無引數,返回一個int型別結果。 |
21 | IntToDoubleFunction 接受一個int型別輸入,返回一個double型別結果 。 |
22 | IntToLongFunction 接受一個int型別輸入,返回一個long型別結果。 |
23 | IntUnaryOperator 接受一個引數同為型別int,返回值型別也為int 。 |
24 | LongBinaryOperator 接受兩個引數同為型別long,返回值型別也為long。 |
25 | LongConsumer 接受一個long型別的輸入引數,無返回值。 |
26 | LongFunction<R> 接受一個long型別輸入引數,返回一個結果。 |
27 | LongPredicate R接受一個long輸入引數,返回一個布林值型別結果。 |
28 | LongSupplier 無引數,返回一個結果long型別的值。 |
29 | LongToDoubleFunction 接受一個long型別輸入,返回一個double型別結果。 |
30 | LongToIntFunction 接受一個long型別輸入,返回一個int型別結果。 |
31 | LongUnaryOperator 接受一個引數同為型別long,返回值型別也為long。 |
32 | ObjDoubleConsumer<T> 接受一個object型別和一個double型別的輸入引數,無返回值。 |
33 | ObjIntConsumer<T> 接受一個object型別和一個int型別的輸入引數,無返回值。 |
34 | ObjLongConsumer<T> 接受一個object型別和一個long型別的輸入引數,無返回值。 |
35 | Predicate<T> 接受一個輸入引數,返回一個布林值結果。 |
36 | Supplier<T> 無引數,返回一個結果。 |
37 | ToDoubleBiFunction<T,U> 接受兩個輸入引數,返回一個double型別結果 |
38 | ToDoubleFunction<T> 接受一個輸入引數,返回一個double型別結果 |
39 | ToIntBiFunction<T,U> 接受兩個輸入引數,返回一個int型別結果。 |
40 | ToIntFunction<T> 接受一個輸入引數,返回一個int型別結果。 |
41 | ToLongBiFunction<T,U> 接受兩個輸入引數,返回一個long型別結果。 |
42 | ToLongFunction<T> 接受一個輸入引數,返回一個long型別結果。 |
43 | UnaryOperator<T> 接受一個引數為型別T,返回值型別也為T。 |
總結
函式式介面 (Functional Interface) 就是一個有且僅有一個抽象方法,但是可以有多個非抽象方法的介面。
函式式介面是為了 lambda 表示式服務,函式式介面的存在是 lambda 表示式出現的前提,lambda 表示式想關於重寫了函式式介面中的唯一方法。