樂字節-Java8核心特性實戰之函數式接口

你們好,上一篇小樂給你們講述了《樂字節-Java8核心特性-Lambda表達式》,點擊回顧。接下來繼續:Java8核心特性之函數式接口。java

何時可使用Lambda?一般Lambda表達式是用在函數式接口上使用的。從Java8開始引入了函數式接口,其說明比較簡單:函數式接口(Functional Interface)就是一個有且僅有一個抽象方法,可是能夠有多個非抽象方法的接口。 java8引入@FunctionalInterface 註解聲明該接口是一個函數式接口。數據庫

1、語法

  • 抽象方法有且僅有一個
  • 接口使用@FunctionalInterface 註解進行標註
  • 接口中能夠存在默認方法和靜態方法實現

以下形式:編程

/**
 * 定義函數式接口
 * 接口上標註@FunctionalInterface 註解
 */
@FunctionalInterface
public interface ICollectionService {
    /**
     * 定義打印方法
     */
    void print();
}

在Java8 之前,已有大量函數式接口形式的接口(接口中只存在一個抽象方法),只是沒有強制聲明。例如java.lang.Runnable,java.util.concurrent.Callable,java.security.PrivilegedAction,java.io.FileFilter等,Java8 新增長的函數接口在java.util.function 包下,它包含了不少類,用來支持 Java的 函數式編程,該包中的函數式接口以下:segmentfault

序號 接口 & 描述
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 LongPredicateR接受一個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。

對於Java8中提供的這麼多函數式接口,開發中經常使用的函數式接口有如下幾個 Predicate,Consumer,Function,Supplier。app

2、函數式接口實例

一、Predicate

java.util.function.Predicate<T> 接口定義了一個名叫 test 的抽象方法,它接受泛型 T 對象,並返回一個boolean值。在對類型 T進行斷言判斷時,可使用這個接口。一般稱爲斷言型接口 。函數式編程

  • 字符串判空
Predicate<String> p01=(str)->str.isEmpty()||str.trim().isEmpty();
     /**
      * 測試傳入的字符串是否爲空
      */
    System.out.println(p01.test(""));
    System.out.println(p01.test("  "));
    System.out.println(p01.test("admin"));
  • 用戶合法性校驗

接口靜態方法完成手機號合法校驗功能,方法返回函數式接口Predicate函數

public interface MyStringInter {
    public final  String checkPhone= "^((13[0-9])|(14[5,7,9])|(15([0-3]|[5-9]))|(16[0-9])" +
            "|(17[0,1,3,5,6,7,8])|(18[0-9])|(19[8|9]))\\d{8}$";
    /**
     * 用戶手機格式合法性
     *     返回L函數式接口Predicate 的實現  Lambda表達式
     * @return
     */
    static Predicate<String> checkPhone(){
          return (e)-> {
              return Pattern.compile(checkPhone).matcher(e).matches();
          };
    }
}

二、Consumer

java.util.function.Consumer<T>接口定義了一個名叫 accept 的抽象方法,它接受泛型T,沒有返回值(void)。若是須要訪問類型 T 的對象,並對其執行某些操做,可使用這個接口,一般稱爲消費型接口。測試

  • 熱銷商品展現
/**
  熱銷商品測試數據
*/
Goods g01=new Goods(1,"iPad 2018款",3000,180, BigDecimal.valueOf(2300));
Goods g02=new Goods(6,"小米平板4",5000,600, BigDecimal.valueOf(1900));
Goods g03=new Goods(9,"微軟 Surface Pro 6",100,50, BigDecimal.valueOf(8500));
Goods g04=new Goods(20,"華爲 榮耀平板5",1600,480, BigDecimal.valueOf(1500));
List<Goods> goods= Arrays.asList(g01,g02,g03,g04);


//Consumer 實現集合數據輸出 Lambda替代匿名函數 實現Consumer接口
 goods.forEach(g->{
     System.out.println(g);
 });

三、Function

​ java.util.function.Function<T, R>接口定義了一個叫做apply的方法,它接受一個泛型T的對象,並返回一個泛型R的對象。若是須要定義一個Lambda,將輸入的信息映射到輸出,可使用這個接口(好比提取蘋果的重量,或把字符串映射爲它的長度),一般稱爲功能型接口。網站

  • 用戶密碼 Base64編碼
// 實現用戶密碼 Base64加密操做
Function<String,String> f01=(password)->Base64.getEncoder().encodeToString(password.getBytes());
// 輸出加密後的字符串
System.out.println(f01.apply("123456"));

四、Supplier

java.util.function.Supplier<T>接口定義了一個get的抽象方法,它沒有參數,返回一個泛型T的對象,這相似於一個工廠方法,一般稱爲功能型接口。編碼

  • 外部Properties文件讀取
public static Properties readFile(String fileName) {
        Supplier<Properties> supplier = () -> {
            try {
                InputStream is = TestCase04.class.getClassLoader().getResourceAsStream(fileName);
                Properties prop = new Properties();
                prop.load(is);
                return prop;
            } catch (IOException e) {
                e.printStackTrace();
                return null;
            }
        };
        return supplier.get();
    }

3、高階函數

圖片描述

Java8 中函數式接口中方法容許函數接口做爲方法形參傳入,同時方法的結果爲函數接口,從而實現鏈式調用操做,就像俄羅斯套娃那樣,當把套娃一個個打開時,發現還有一個一樣的小套娃在裏面,最終發現最裏面的一個也是一個完整的套娃玩具,此時的高階函數是否是跟套娃有着驚人的類似之處呢。

  • 多頁面轉發
String action = "";
Predicate<String> p01 = (a) -> StringUtils.isBlank(a);
/**
* 若是action 爲空  或 index  或 main 轉發到網站主頁面
* 鏈式判斷 方法結果仍然爲一個函數
*/
if (p01.or((a) -> a.equals("index")).or((a) -> a.equals("main")).test(action)) {
    System.out.println("網站主頁面...");
} else {
    System.out.println("其餘頁面...");
}
  • 多條件排序

這裏以商品數據爲例,按商品銷量、評論排序,若是銷量一致 按照商品評論數排序

/**
* 實際開發數據一般從數據庫獲取
* 這裏使用測試數據
*/
Goods g01=new Goods(1,"小米9",1789,200, BigDecimal.valueOf(2500));
Goods g02=new Goods(2,"華爲Mate20",5000,3000, BigDecimal.valueOf(7000));
Goods g03=new Goods(3,"OPPO R17",2000,2827, BigDecimal.valueOf(1500));
Goods g04=new Goods(4,"魅族 Note9",2000,1600, BigDecimal.valueOf(1600));
Goods g05=new Goods(5,"一加6T",8000,5000, BigDecimal.valueOf(3500));
List<Goods> goods= Arrays.asList(g01,g02,g03,g04,g05);


// 銷量 與 評論排序 高階函數使用
Comparator<Goods> comparator = (g1,g2)->g1.getSale()-g2.getSale();
goods.sort(comparator.thenComparing(Comparator.comparing(g3 -> g3.getComment())));
goods.forEach((g)->System.out.println(g));

高階函數應用場景較多(這裏查看源碼相關高階函數方法),如Optinal接口filter、map 、orElseGet 等方法,Stream 流操做等 基本都會用到 Predicate,Consumer,Supplier,Function等接口。

4、函數式接口優點與應用場景

函數式接口的引入,結合Lambda的使用,消除的匿名函數繁瑣的代碼,使得代碼結構簡潔、緊湊,第二點就是函數式接口中使用高階函數,能夠很方便的實現鏈式調用,代碼清晰簡潔,同時引入的一種新的開發思想-函數式編程,對於開發者來講只須要關注函數的規則設計實現便可。

對於函數式接口應用,後續介紹到的Optinal、Stream相關方法對於數據處理的使用頻率較高,同時也是構成函數式編程的核心內容。

感謝你們欣賞小樂帶來的Java8核心特性之函數式接口,接下來還會更多Java8-Java12核心技術講解,請關注 樂字節 如須要視頻課程,請搜索 樂字節騰訊課堂

相關文章
相關標籤/搜索