1. 程式人生 > >Java 8 新增函式式介面到底是什麼?

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 表示式想關於重寫了函式式介面中的唯一方法。