Java8 -Lambda表達式(6)

1.復合Lambda表達式的有用方法

??Java8的好幾個函數式接口都有為方便而設計的方法。具體而言,許多函數式接口,比如用于傳遞Lambda表達式的Comparator、Function和Predicate都提供了允許你進行復合的方法。這是什么意思呢?在實踐中,這意味著你可以把多個簡單的Lambda復合成為復雜的表達式。比如,你可以讓兩個謂詞之間做一個or操作,組合成為了一個更大的謂詞。而且,你還可以讓一個函數的結果成為另一個函數的輸入。你可能會在想,函數式接口中怎么可能會有更多的方法呢?(畢竟,這違背了函數式接口的定義啊!)。關鍵在于,我們即將介紹的方法都是默認方法,也就是說它們不是抽象方法。

(1).比較器復合

??我們前面看到了,你可以使用Comparator.comparing,根據提取用于比較的鍵值的Function來返回一個Comparator對象,如下所示:

inventory.sort(Comparator.comparing(Apple::getWeight));
A.逆序

??如果你想要對蘋果按質量遞減排序怎么辦?用不著去建立另一個Comparator的實例。接口有一個默認方法reversed可以使給定的比較器排序。因此仍然用開始的那個比較器,只要修改一下前一個例子就可以對蘋果按重量遞減排序:

inventory.sort(Comparator.comparing(Apple::getWeight).reversed());
B.比較器鏈

??上面說的都好,但是如果發現有兩個蘋果一樣重的怎么辦?哪個蘋果應該排在前面呢?你可能需要在提供一個Comparator進來進一步定義這一個比較。比如,在按重量比較兩個蘋果之后,你可能想要按顏色排序。thenComparing方法就是用來做來這個的。它接收一個函數作為參數(就像comparing方法一樣)。如果兩個對象用第一個Comparator比較之后是一樣的,就提供第二個Comparator。你又可以優雅的解決這個問題:

 inventory.sort(Comparator.comparing(Apple::getWeight)
               .reversed() //按照重量遞減排序
              .thenComparing(Apple::getColor)); //兩個蘋果一樣重時,進一步按顏色排序

(2).謂詞復合

??謂詞接口包括三個方法:negate、and和or,讓你可以重用已有的Predicate來創建更加復雜的謂詞。比如你已使用negate方法來返回一個Predicate的非,比如蘋果不是紅的:

Predicate<Apple> redApple = a->a.getWeight().equals("red");
Predicate<Apple> notRedApple = redApple.negate();

??你可能想要把兩個Lambda用and方法組合起來,比如一個蘋果既是紅色又比較中:

Predicate<Apple> redAndHeavyApple = redApple.and(a -> a.getWeight() > 150);

??你可以進一步組合謂詞,表達要么是重(150g以上)的紅蘋果,要么是綠蘋果:

Predicate<Apple> redAndHeavyApple = 
            redApple.and(a -> a.getWeight() > 150) 
           .or(a -> "green".equals(a.getColor()));//鏈接Predicate的方法來構造更加復雜Predicate對象

??這一點為什么很好呢?從簡單Lambda表達his出發,你可以構建更加復雜的表達式,但是讀起來仍然和問題的陳述差不多!請注意,and和or是按照在表達式鏈中的位置,從左往右確定優先級的。因此,a.or(b).and(c)可以看做(a || b) && c。

(3).函數復合

??最后,你可以把Function接口代表的Lambda表達式復合起來。Function接口為此配了andThen和compose兩個默認方法,它們都會返回Function的一個實例。
??andThen方法會返回一個函數,它對輸入應用到一個給定函數,再對輸出應用到另一個函數。比如,假設有一個函數f數字加1(x -> x + 1),另一個函數給g給數字乘以2,你可以將它們組合成一個函數h,先給數字加1,再給結果乘以2。

Function<Integer, Integer> f = x -> x + 1;
Function<Integer, Integer> g = x -> x * 2;
Function<Integer, Integer> h = f.andThen(g); //數學上寫作g(f(x))
int result = h.apply(1); //返回的是4

??你可以類似的使用compose方法,先把給定的函數用作compose的參數里面給的那個函數,然后再把函數本身用于結果。比如上一個例子用compose的話,它將意味著f(g(x)),而andThen則意味著g(f(x)):

Function<Integer, Integer> f = x -> x + 1;
Function<Integer, Integer> g = x -> x * 2;
Function<Integer, Integer> h = f.compose(g); //數學上寫作f(g(x))
int result = h.apply(1); //返回的是3

??andThen和compose的區別:



??這一切聽起來有點太抽象了。那么在實際中這有什么用呢?比方說,你有一系列工具方法,對用String表示的一封信做文本轉換:

public class Letter {
    public static String addHeader(String text){
        return "From pby" + text;
    }
    public static String addFooter(String text){
        return text + "Kind regards";
    }
    public static String checkSpelling(String text){
        return text.replaceAll("labda", "lambda");
    }
}

??現在你可以通過復合這些工具方法來創建各種轉型流水線了,比如創建一個流水線:先加上頭部,然后進行拼寫檢查,最后加上一個落款。如圖所示:


 Function<String ,String> addHeader = Letter::addHeader;
Function<String, String> transformationPipeline = 
                 addHeader.andThen(Letter::checkSpelling)
                 .andThen(Letter::addFooter);

??第二個流水線可能只加頭部和落款,而不做拼寫檢查:

Function<String ,String> addHeader = Letter::addHeader;
Function<String ,String> transformationPipeline = addHeader.andThen(Letter::addFooter);
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容

  • 簡介 概念 Lambda 表達式可以理解為簡潔地表示可傳遞的匿名函數的一種方式:它沒有名稱,但它有參數列表、函數主...
    劉滌生閱讀 3,222評論 5 18
  • Lambda表達式 利用行為參數化這個概念,就可以編寫更為靈活且可重復使用的代碼。但同時,使用匿名類來表示不同的行...
    謝隨安閱讀 884評論 2 0
  • Java8 in action 沒有共享的可變數據,將方法和函數即代碼傳遞給其他方法的能力就是我們平常所說的函數式...
    鐵牛很鐵閱讀 1,259評論 1 2
  • 第一章 為什么要關心Java 8 使用Stream庫來選擇最佳低級執行機制可以避免使用Synchronized(同...
    謝隨安閱讀 1,508評論 0 4
  • 他或許明白,這罪惡的時代,罪惡的人生,發起狂來,便似瘋狗,逢人就咬,半點兒情面不留,果然吶果然! 往日里咕嚕哇呀的...
    客從遠方來_閱讀 360評論 0 3