偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

Java8 函數(shù)式方法引用優(yōu)秀實(shí)踐

開(kāi)發(fā)
本文我們將 Java8 函數(shù)式方法引用的推導(dǎo)和實(shí)用,以及各種表達(dá)式組合的內(nèi)容都會(huì)介紹到,希望對(duì)你有幫助。

一、詳解lambda中的方法引用

1. 方法引用使用的推導(dǎo)

我們現(xiàn)在有一個(gè)蘋(píng)果類,其代碼定義如下:

@Data
@AllArgsConstructor
public class Apple  {
    private int weight;
    
}

因?yàn)橹亓繂挝坏牟煌?,所以得出的重量的結(jié)果可能是不同的,所以我們將計(jì)算重量的核心部分抽象成函數(shù)式接口,如下function所示,它要求我們傳入Apple返回Integer:

private static int getWeight(Apple apple, Function<Apple,Integer> function) {
        return function.apply(apple);
    }

假設(shè)我們對(duì)重量無(wú)需任何單位換算即原原本本返回重量本身,那么我們的表達(dá)式則直接是(a)->a.getWeight(),對(duì)應(yīng)代碼如下:

Apple apple=new Apple(1);
 System.out.println(getWeight(apple,(a)->a.getWeight()));

其實(shí)這個(gè)表達(dá)式還不是最精簡(jiǎn)的,按照方法引用的語(yǔ)法糖,如果我們的lambda表達(dá)式符合:(arg)->arg.method(),即傳入的lambda就是(實(shí)例變量)->實(shí)例變量.實(shí)例方法(),那么這個(gè)表達(dá)式就可以直接縮寫(xiě)為arg ClassName::invokeMethod:

于是我們的代碼就可以精簡(jiǎn)成下面這樣:

System.out.println(getWeight(apple,Apple::getWeight));

除了上述這個(gè)公式以外,其實(shí)還有另外兩種公式,如下所示我們的map映射希望將流中的字符串轉(zhuǎn)為整型,然后輸出:

Arrays.asList("1").stream()
                .map(s -> Integer.parseInt(s))
                .forEach(i -> System.out.println(i));

按照jdk8的語(yǔ)法糖,對(duì)應(yīng)的靜態(tài)類調(diào)用靜態(tài)方法的表達(dá)式(args)->className.staticMethod(args)可以直接縮寫(xiě)為className->staticMethod(args),于是我們的整型轉(zhuǎn)換的就可以直接縮寫(xiě)為Integer::parseInt:

Arrays.asList("1").stream()
                .map(Integer::parseInt)
                .forEach(i -> System.out.println(i));

最后一種則是針對(duì)多參數(shù)的如下所示,這是一個(gè)常規(guī)的排序lambda編程:

List<String> str = Arrays.asList("a","b","A","B");
str.sort((s1, s2) -> s1.compareToIgnoreCase(s2));

按照J(rèn)ava8的語(yǔ)法糖:(arg1,arg2)->arg1.instanceMethod(arg2)可以直接轉(zhuǎn)換為arg1ClassName::invokeInstanceMethod,于是我們的就有了下面的推導(dǎo):

最終我們的表達(dá)式就變成了這樣:

List<String> str = Arrays.asList("a","b","A","B");
        str.sort(String::compareToIgnoreCase);

2. 方法引用對(duì)于對(duì)象構(gòu)造的抽象

實(shí)際上對(duì)象構(gòu)造也可以通過(guò)方法引用表達(dá),其整體縮寫(xiě)的語(yǔ)法和靜態(tài)方法引用類似,如下圖所示本質(zhì)上new的動(dòng)作就可以直接理解為對(duì)于new的調(diào)用,同理簡(jiǎn)寫(xiě)為className::new來(lái)表達(dá):

我們不妨結(jié)合幾個(gè)例子進(jìn)行說(shuō)明,如下便是蘋(píng)果對(duì)象的類定義,即帶有重量、顏色等屬性,同時(shí)支持含參或不含參的方式構(gòu)造:

@Data
@AllArgsConstructor
@NoArgsConstructor
public class Apple {
    private int weight;

    private String color;


}

簡(jiǎn)單的蘋(píng)果對(duì)象創(chuàng)建就像下面這樣new創(chuàng)建

//普通對(duì)象創(chuàng)建
        Apple apple = new Apple();

實(shí)際上這個(gè)創(chuàng)建步驟在函數(shù)式中可以抽象的理解為Supplier接口()->T,其中T為Apple,所以我們表達(dá)式可以轉(zhuǎn)換為如下方式:

Supplier<Apple> apply = () -> new Apple();
 Apple apple1 = apply.get();

此時(shí),基于我們上述的圖解,即可將Supplier對(duì)象構(gòu)造推導(dǎo)出構(gòu)造函數(shù)的方法引用:

于是就有了下方代碼:

//采用方法引用縮寫(xiě)
        Supplier<Apple> apply2 = Apple::new;
        Apple apple2 = apply2.get();

我們?cè)賮?lái)一個(gè)難一點(diǎn)的例子,因?yàn)槲覀兊臉?gòu)造器為傳參順序?yàn)閣eight、color然后創(chuàng)建Apple實(shí)例,對(duì)此我們可以大體抽象出函數(shù)式接口的簽名為(Integer,String)->Apple,基于這個(gè)簽名我們可以直接套用公式BiFunction,它的簽名為(T,U)->R,參數(shù)列表符合要求,我們直接將類型代入完成函數(shù)式接口抽象:

private static Apple createApple(Integer weight,String color,BiFunction<Integer, String, Apple> func) {
        return func.apply(weight, color);
    }

基于上述的簽名的參數(shù)列表和預(yù)期返回值,我們得出下面這樣一條lambda表達(dá)式作為入?yún)魅?,由此得到一個(gè)Apple實(shí)例:

createApple(1,"yellow",(w,s)->new Apple(w,s));

按照上文所說(shuō)的公式,于是我們的表達(dá)式又可以轉(zhuǎn)為方法引用:

對(duì)應(yīng)的代碼如下所示:

createApple(1,"yellow",Apple::new);

3. lambda和方法引用的結(jié)合

我們希望對(duì)蘋(píng)果類進(jìn)行排序,對(duì)此我們給出蘋(píng)果類的實(shí)例集合:

List<Apple> appleList = Arrays.asList(new Apple(80, "green"),
                new Apple(200, "red"),
                new Apple(155, "yellow"),
                new Apple(120, "red"));

查看函數(shù)式接口Comparator的抽象方法 int compare(T o1, T o2);得出對(duì)應(yīng)的函數(shù)簽名為(T,T)->Integer,代入我們的Apple類,那么這個(gè)比較器的函數(shù)描述符則是(Apple,Apple)->Integer,于是我們就有了下面這條lambda表達(dá)式:

Comparator<Apple> comparator = (a1,a2)->a1.getWeight()-a2.getWeight();

我們鍵入如下代碼進(jìn)行調(diào)用輸出:

appleList.sort(comparator);
 appleList.forEach(System.out::println);

和預(yù)期比較結(jié)果一致:

Apple(weight=80, color=green)
Apple(weight=120, color=red)
Apple(weight=155, color=yellow)
Apple(weight=200, color=red)

實(shí)際上我們還可以做的更加精簡(jiǎn),因?yàn)镴DK8中的Comparator已經(jīng)為比較器提供了一個(gè)方法comparing,查看其源碼可以看到他要求傳入一個(gè)入?yún)eyExtractor,從語(yǔ)義上就可以知道這個(gè)參數(shù)是作為比較的條件,以我們的例子就是Apple的weight。 這個(gè)keyExtractor是Function接口,查看其泛型我們也可以知曉它的函數(shù)式簽名為T(mén)->R,由此我們可以推理出該方法本質(zhì)就是通過(guò)Function接口變量keyExtractor生成比較變量的實(shí)例然后調(diào)用compareTo進(jìn)行比較并返回結(jié)果:

//要求傳入keyExtractor即作為比較的條件
public static <T, U extends Comparable<? super U>> Comparator<T> comparing(
            Function<? super T, ? extends U> keyExtractor)
    {
        //......
        return (Comparator<T> & Serializable)
         //通過(guò)keyExtractor生成key值調(diào)用其compareTo方法進(jìn)行比較
            (c1, c2) -> keyExtractor.apply(c1).compareTo(keyExtractor.apply(c2));
    }

基于上述分析我們就可以開(kāi)始編寫(xiě)這個(gè)比較器的keyExtractor的lambda表達(dá)式了,如下圖,通過(guò)keyExtractor泛型得出函數(shù)描述符為(T)->R,基于我們的場(chǎng)景推導(dǎo)出公式是apple實(shí)例->apple實(shí)例的weight,最后comparing會(huì)基于這個(gè)函數(shù)接口生成的R對(duì)象(我們的場(chǎng)景是weight即int類型)調(diào)用compareTo進(jìn)行比較:

于是我們就有了這樣一條lambda表達(dá)式,但這還不是最精簡(jiǎn)的:

Comparator<Apple> comparator = Comparator.comparing(a->a.getWeight());

按照l(shuí)ambda的語(yǔ)法糖:instance->instance.method 可以直接轉(zhuǎn)為instanceType::method,我們最終的表達(dá)式如下,預(yù)期結(jié)果也和之前一致:

Comparator<Apple> comparator = Comparator.comparing(Apple::getWeight);

當(dāng)然有時(shí)候我們希望能夠?qū)Y(jié)果進(jìn)行反向排序,我們也只需在comparing方法后面加一個(gè)reversed即實(shí)現(xiàn),從語(yǔ)義和使用上是不是都很方便呢?

Comparator<Apple> comparator = Comparator.comparing(Apple::getWeight).reversed();

二、復(fù)合表達(dá)式

1. 復(fù)合比較器

自此我們基本將方法引用的推導(dǎo)和使用都講完了,接下來(lái)我們還是基于lambda做一些實(shí)用的拓展,先來(lái)說(shuō)說(shuō)復(fù)合比較器,以上文的蘋(píng)果為例,假設(shè)我們希望當(dāng)重量一樣時(shí),在比較顏色進(jìn)行進(jìn)一步比較,那么我們就可以直接通過(guò)thenComparing生成復(fù)合表達(dá)式:

Comparator<Apple> comparator = Comparator.comparing(Apple::getWeight).reversed().thenComparing(Apple::getColor);

2. 謂詞復(fù)合

還是用上面的例子,我們希望根據(jù)不同的條件從蘋(píng)果集合中過(guò)濾出復(fù)合條件的蘋(píng)果,對(duì)此我們基于Predicate即斷言函數(shù)式接口編寫(xiě)了一個(gè)filterApple方法:

private static List<Apple> filterApple(List<Apple> appleList, Predicate<Apple> predicate) {
        List<Apple> list = new ArrayList<>();
        for (Apple apple : appleList) {
            //復(fù)合predicate設(shè)定條件的蘋(píng)果存入集合中
            if (predicate.test(apple)) {
                list.add(apple);
            }
        }
        return list;
    }

假如客戶需要過(guò)濾出紅色的蘋(píng)果,基于predicate的簽名我們得出這樣一個(gè)表達(dá)式,這里就不多介紹了:

filterApple(appleList, apple -> apple.getColor().equals("red"));

假如這時(shí)候我們有需要過(guò)濾出不為紅色的蘋(píng)果呢?其實(shí)JDK8為我們提供了一個(gè)非常強(qiáng)大的謂詞negate,我們完全可以基于上面的代碼進(jìn)行改造從而實(shí)現(xiàn)需求,如下所示negate就相當(dāng)于!"red".equals(a.getColor());,語(yǔ)義是不是很清晰呢?

Predicate<Apple> predicate = apple -> apple.getColor().equals("red");
        filterApple(appleList, predicate.negate());

但是我們需要再次變化了,我們希望找出紅色且重量大于150,或者顏色為綠色的蘋(píng)果,這時(shí)候又怎么辦呢?我們說(shuō)過(guò)JDK8提供了and、or等謂詞,我們的代碼完全可以寫(xiě)成下文所示,可以看到代碼語(yǔ)義以及流暢度都相比JDK8之前的各種&& ||拼接for循環(huán)來(lái)說(shuō)優(yōu)雅非常多:

//過(guò)濾出紅色的蘋(píng)果
        Predicate<Apple> predicate = apple -> apple.getColor().equals("red");
        //過(guò)濾出紅色且大于150 或者綠色的蘋(píng)果
        Predicate<Apple> redAndHeavyAppleOrGreen = predicate.and(apple -> apple.getWeight() > 150).
                or(apple -> apple.getColor().equals("green"));


        filterApple(appleList, redAndHeavyAppleOrGreen);

3. 函數(shù)復(fù)合

我們都說(shuō)代碼和數(shù)學(xué)息息相關(guān),其實(shí)java8也提供很多函數(shù)式接口可以運(yùn)用于數(shù)學(xué)公式上,例如,我們現(xiàn)在需要計(jì)算f(g(x)),這個(gè)公式學(xué)過(guò)高數(shù)的同學(xué)都知道,是先計(jì)算g(x)再將g(x)的結(jié)果作為入?yún)⒔唤of(x)計(jì)算,對(duì)應(yīng)題解案例如下:

我們假設(shè)g(x)=x * 2
f(x)=x+1
假如x=1
那么g(f(x))最終就會(huì)等于4

了解數(shù)學(xué)公式之后,我們完全可以使用java代碼表示出來(lái),首先我們先聲明一下f(x)和g(x):

//f(x)
 Function<Integer, Integer> f = x -> x + 1;
 //g(x)
 Function<Integer, Integer> g = x -> x * 2;

在表示g(f(x)),通過(guò)復(fù)合表達(dá)式andThen表達(dá)了數(shù)學(xué)的計(jì)算順序,即顯得出f(x)結(jié)果,然后(andThen)代入g(x)中:

//意味先計(jì)算f(x)在計(jì)算g(x)
 Function<Integer, Integer> h = f.andThen(g);
System.out.println(result); //輸出 4

基于上面的例子,如果我們還需要計(jì)算f(g(x))要怎么辦呢?從f(x)角度來(lái)看,g(x)的結(jié)果組合到f(x)上,所以我們可以直接實(shí)用compose方法:

Function<Integer, Integer> fgx = f.compose(g);
Integer result = fgx.apply(1);
System.out.println(result);// 輸出 3

其實(shí),按照奧卡姆剃刀守則,如果按照筆者的習(xí)慣,會(huì)優(yōu)先使用第一種,即fg(x)用 g.andThen(f);,即先算g再算f,而gf(x)則用f.andThen(g);即先算f再算g。

責(zé)任編輯:趙寧寧 來(lái)源: 寫(xiě)代碼的SharkChili
相關(guān)推薦

2015-09-30 09:34:09

java8字母序列

2020-05-25 16:25:17

Java8Stream函數(shù)式接口

2023-07-26 07:13:55

函數(shù)接口Java 8

2019-01-17 10:25:56

Python編程語(yǔ)言程序員

2022-12-01 07:38:49

lambda表達(dá)式函數(shù)式

2020-10-16 10:07:03

Lambda表達(dá)式Java8

2021-03-04 08:14:37

Java8開(kāi)發(fā)接口

2023-06-08 16:47:09

軟件開(kāi)發(fā)工具

2022-12-26 07:47:37

JDK8函數(shù)式接口

2023-10-19 08:00:00

2019-04-26 07:56:40

容器秘密安全

2014-09-05 10:15:41

函數(shù)式編程

2022-11-03 08:16:33

MySQL·窗口函數(shù)

2024-09-29 15:21:01

2023-01-09 11:45:21

Java8Optional系統(tǒng)

2010-06-22 13:32:26

函數(shù)式編程JavaScript

2020-01-15 15:12:38

Java8日期處理代碼

2020-02-06 11:35:58

Java 8APIJava

2025-06-12 09:48:46

2019-11-05 17:10:19

Java開(kāi)發(fā)編程語(yǔ)言
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)