成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Java 8th 函數(shù)式編程:lambda 表達(dá)式

luffyZh / 730人閱讀

摘要:自定義函數(shù)式接口我們在前面例子中實(shí)現(xiàn)的蘋果篩選接口就是一個函數(shù)式接口定義如下,正因?yàn)槿绱宋覀兛梢詫⒑Y選邏輯參數(shù)化,并應(yīng)用表達(dá)式僅包含一個抽象方法,依照定義可以將其視為一個函數(shù)式接口。

Lambda 表達(dá)式是 java 8th 給我們帶來的幾個重量級新特性之一,借用 lambda 表達(dá)式可以讓我們的程序設(shè)計(jì)更加簡潔。最近新的項(xiàng)目摒棄了 6th 版本,全面基于 8th 進(jìn)行開發(fā),本文將探討 行為參數(shù)化lambda 表達(dá)式 , 以及 方法引用 等知識點(diǎn)。

一. 行為參數(shù)化

行為參數(shù)化簡單的說就是將方法的邏輯以參數(shù)的形式傳遞到方法中,方法主體僅包含模板類通用代碼,而一些會隨著業(yè)務(wù)場景而變化的邏輯則以參數(shù)的形式傳遞到方法之中,采用行為參數(shù)化可以讓程序更加的通用,以應(yīng)對頻繁變更的需求。

這里我們以 java 8 in action 中的例子進(jìn)行說明??紤]一個業(yè)務(wù)場景,假設(shè)我們需要通過程序?qū)μO果按照一定的條件進(jìn)行篩選,我們先定義一個蘋果實(shí)體:

public class Apple {
    /** 編號 */
    private Long id;
    /** 顏色 */
    private Color color;
    /** 重量 */
    private Float weight;
    /** 產(chǎn)地 */
    private String origin;

    public Apple() {
    }

    public Apple(Long id, Color color, Float weight, String origin) {
        this.id = id;
        this.color = color;
        this.weight = weight;
        this.origin = origin;
    }

    // 省略getter和setter
}

用戶最開始的需求可能只是簡單的希望能夠通過程序篩選出綠色的蘋果,于是我們可以很快的通過程序?qū)崿F(xiàn):

public static List filterGreenApples(List apples) {
    List filterApples = new ArrayList<>();
    for (final Apple apple : apples) {
        // 篩選出綠色的蘋果
        if (Color.GREEN.equals(apple.getColor())) {
            filterApples.add(apple);
        }
    }
    return filterApples;
}

如果過了一段時(shí)間用戶提出了新的需求,希望能夠通過程序篩選出紅色的蘋果,于是我們又需要針對性的添加了篩選紅色蘋果的功能:

public static List filterRedApples(List apples) {
    List filterApples = new ArrayList<>();
    for (final Apple apple : apples) {
        // 篩選出紅色的蘋果
        if (Color.RED.equals(apple.getColor())) {
            filterApples.add(apple);
        }
    }
    return filterApples;
}

更通用的實(shí)現(xiàn)是把顏色作為一個參數(shù)傳遞到方法中,這樣就可以應(yīng)對以后用戶提出的各種顏色篩選需求:

public static List filterApplesByColor(List apples, Color color) {
    List filterApples = new ArrayList<>();
    for (final Apple apple : apples) {
        // 依據(jù)傳入的顏色參數(shù)進(jìn)行篩選
        if (color.equals(apple.getColor())) {
            filterApples.add(apple);
        }
    }
    return filterApples;
}

這樣的設(shè)計(jì)再也不用擔(dān)心用戶的顏色篩選需求變化了,但是不幸的是某一天用戶提了一個需求希望能夠篩選重量達(dá)到某一標(biāo)準(zhǔn)的蘋果,有了前面的教訓(xùn)我們也把重量的標(biāo)準(zhǔn)作為參數(shù)傳遞給篩選函數(shù):

public static List filterApplesByColorAndWeight(List apples, Color color, float weight) {
    List filterApples = new ArrayList<>();
    for (final Apple apple : apples) {
        // 依據(jù)顏色和重量進(jìn)行篩選
        if (color.equals(apple.getColor()) && apple.getWeight() >= weight) {
            filterApples.add(apple);
        }
    }
    return filterApples;
}

這樣通過傳遞參數(shù)的方式真的好嗎?如果篩選條件越來越多,組合模式越來越復(fù)雜,我們是不是需要考慮到所有的情況,并針對每一種情況都實(shí)現(xiàn)相應(yīng)的策略呢?并且這些函數(shù)僅僅是篩選條件的部分不一樣,其余部分都是相同的模板代碼(遍歷集合),這個時(shí)候我們就可以將行為進(jìn)行 參數(shù)化 處理,讓函數(shù)僅保留模板代碼,而把篩選條件抽離出來當(dāng)做參數(shù)傳遞進(jìn)來,在 java 8th 之前,我們通過定義一個過濾器接口來實(shí)現(xiàn):

// 過濾器
public interface AppleFilter {
    boolean accept(Apple apple);
}

// 應(yīng)用過濾器的篩選方法
public static List filterApplesByAppleFilter(List apples, AppleFilter filter) {
    List filterApples = new ArrayList<>();
    for (final Apple apple : apples) {
        if (filter.accept(apple)) {
            filterApples.add(apple);
        }
    }
    return filterApples;
}

通過上面行為抽象化之后,我們可以在具體調(diào)用的地方設(shè)置篩選條件,并將條件作為參數(shù)傳遞到方法中:

public static void main(String[] args) {
    List apples = new ArrayList<>();

    // 篩選蘋果
    List filterApples = filterApplesByAppleFilter(apples, new AppleFilter() {
        @Override
        public boolean accept(Apple apple) {
            // 篩選重量大于100g的紅蘋果
            return Color.RED.equals(apple.getColor()) && apple.getWeight() > 100;
        }
    });
}

上面的行為參數(shù)化方式采用匿名類實(shí)現(xiàn),這樣的設(shè)計(jì)在 jdk 內(nèi)部也經(jīng)常采用,比如 java.util.Comparatorjava.util.concurrent.Callable 等,使用這類接口的時(shí)候,我們都可以在具體調(diào)用的地方用匿名類指定函數(shù)的具體執(zhí)行邏輯,不過從上面的代碼塊來看,雖然很極客,但是不夠簡潔,在 java 8th 中我們可以通過 lambda 表達(dá)式進(jìn)行簡化:

// 篩選蘋果
List filterApples = filterApplesByAppleFilter(apples,
        (Apple apple) -> Color.RED.equals(apple.getColor()) && apple.getWeight() >= 100);

如上述所示,通過 lambda 表達(dá)式極大精簡了代碼,同時(shí)行為參數(shù)讓我們的程序極大的增強(qiáng)了可擴(kuò)展性。

二. Lambda 表達(dá)式 2.1 Lambda 表達(dá)式的定義與形式

我們可以將 lambda 表達(dá)式定義為一種 __簡潔、可傳遞的匿名函數(shù)__,首先我們需要明確 lambda 表達(dá)式本質(zhì)上是一個函數(shù),雖然它不屬于某個特定的類,但具備參數(shù)列表、函數(shù)主體、返回類型,甚至能夠拋出異常;其次它是匿名的,lambda 表達(dá)式?jīng)]有具體的函數(shù)名稱;lambda 表達(dá)式可以像參數(shù)一樣進(jìn)行傳遞,從而簡化代碼的編寫,其格式定義如下:

參數(shù)列表 -> 表達(dá)式

參數(shù)列表 -> {表達(dá)式集合}

需要注意 lambda 表達(dá)式隱含了 return 關(guān)鍵字,所以在單個的表達(dá)式中,我們無需顯式的寫 return 關(guān)鍵字,但是當(dāng)表達(dá)式是一個語句集合的時(shí)候則需要顯式添加 return 關(guān)鍵字,并用花括號 {} 將多個表達(dá)式包圍起來,下面看幾個例子:

// 1. 返回給定字符串的長度(隱含return語句)
(String s) -> s.length()

// 2. 始終返回42的無參方法(隱含return語句)
() -> 42

// 3. 包含多行表達(dá)式,需用花括號括起來,并顯示添加return
(int x, int y) -> {
    int z = x * y;
    return x + z;
}
2.2 基于函數(shù)式接口使用 lambda 表達(dá)式

lambda 表達(dá)式的使用需要借助于 __函數(shù)式接口__,也就是說只有函數(shù)式接口出現(xiàn)地方,我們才可以將其用 lambda 表達(dá)式進(jìn)行簡化。那么什么是函數(shù)接口?函數(shù)接口的定義如下:

函數(shù)式接口定義為僅含有一個抽象方法的接口。

按照這個定義,我們可以確定一個接口如果聲明了兩個或兩個以上的方法就不叫函數(shù)式接口,需要注意一點(diǎn)的是 java 8th 為接口的定義引入了默認(rèn)的方法,我們可以用 default 關(guān)鍵字在接口中定義具備方法體的方法,這個在后面的文章中專門講解,如果一個接口存在多個默認(rèn)方法,但是仍然僅含有一個抽象方法,那么這個接口也符合函數(shù)式接口的定義。

2.2.1 自定義函數(shù)式接口

我們在前面例子中實(shí)現(xiàn)的蘋果篩選接口就是一個函數(shù)式接口(定義如下),正因?yàn)槿绱宋覀兛梢詫⒑Y選邏輯參數(shù)化,并應(yīng)用 lambda 表達(dá)式:

@FunctionalInterface
public interface AppleFilter {
    boolean accept(Apple apple);
}

AppleFilter 僅包含一個抽象方法 accept(Apple apple),依照定義可以將其視為一個函數(shù)式接口。在定義時(shí)我們?yōu)樵摻涌谔砑恿?@FunctionalInterface 注解,用于標(biāo)記該接口是一個函數(shù)式接口,不過該注解是可選的,當(dāng)添加了該注解之后,編譯器會限制了該接口只允許有一個抽象方法,否則報(bào)錯,所以推薦為函數(shù)式接口添加該注解。

2.2.2 jdk 自帶的函數(shù)式接口

jdk 為 lambda 表達(dá)式已經(jīng)內(nèi)置了豐富的函數(shù)式接口,如下表所示(僅列出部分):

函數(shù)式接口 函數(shù)描述符 原始類型特化
Predicate T -> boolean IntPredicate, LongPredicate, DoublePredicate
Consumer T -> void IntConsumer, LongConsumer, DoubleConsumer
Funcation T -> R IntFuncation, IntToDoubleFunction, IntToLongFunction, LongFuncation...
Supplier () -> T BooleanSupplier, IntSupplier, LongSupplier, DoubleSupplier
UnaryOperator T -> T IntUnaryOperator, LongUnaryOperator, DoubleUnaryOperator
BinaryOperator (T, T) -> T IntBinaryOperator, LongBinaryOperator, DoubleBinaryOperator
BiPredicate (L, R) -> boolean
BiConsumer (T, U) -> void
BiFunction (T, U) -> R

其中最典型的三個接口是 PredicateConsumer,以及 Function,其余接口幾乎都是對這三個接口的定制化,下面就這三個接口舉例說明其用處,針對接口中提供的邏輯操作默認(rèn)方法,留到后面介紹接口的 default 方法時(shí)再進(jìn)行說明。

Predicate

@FunctionalInterface
public interface Predicate {

    /**
     * Evaluates this predicate on the given argument.
     *
     * @param t the input argument
     * @return {@code true} if the input argument matches the predicate,
     * otherwise {@code false}
     */
    boolean test(T t);
}

Predicate 的功能類似于上面的 AppleFilter,利用我們在外部設(shè)定的條件對于傳入的參數(shù)進(jìn)行校驗(yàn)并返回驗(yàn)證通過與否,下面利用 Predicate 對 List 集合的元素進(jìn)行過濾:

private  List filter(List numbers, Predicate predicate) {
    Iterator itr = numbers.iterator();
    while (itr.hasNext()) {
        if (!predicate.test(itr.next())) {
            itr.remove();
        }
        itr.next();
    }
    return numbers;
}

上述方法的邏輯是遍歷集合中的元素,通過 Predicate 對集合元素進(jìn)行驗(yàn)證,并將驗(yàn)證不過的元素從集合中移除。我們可以利用上面的函數(shù)式接口篩選整數(shù)集合中的偶數(shù):

PredicateDemo pd = new PredicateDemo();
List list = new ArrayList<>();
list.addAll(Arrays.asList(1, 2, 3, 4, 5, 6));
list = pd.filter(list, (value) -> value % 2 == 0);
System.out.println(list);
// 輸出:[2, 4, 6]

Consumer

@FunctionalInterface
public interface Consumer {

    /**
     * Performs this operation on the given argument.
     *
     * @param t the input argument
     */
    void accept(T t);
}

Consumer 提供了一個 accept 抽象函數(shù),該函數(shù)接收參數(shù)并依據(jù)傳遞的行為應(yīng)用傳遞的參數(shù)值,下面利用 Consumer 遍歷字符串集合并轉(zhuǎn)換成小寫進(jìn)行打?。?/p>

private  void forEach(List list, Consumer consumer) {
    for (final T value : list) {
        // 應(yīng)用行為
        consumer.accept(value);
    }
}

利用上面的函數(shù)式接口,遍歷字符串集合并以小寫形式打印輸出:

ConsumerDemo cd = new ConsumerDemo();
List list = new ArrayList<>();
list.addAll(Arrays.asList("I", " ", "Love", " ", "Java", " ", "8th"));
cd.forEach(list, (value) -> System.out.print(value.toLowerCase()));
// 輸出:i love java 8th

Function

@FunctionalInterface
public interface Function {

    /**
     * Applies this function to the given argument.
     *
     * @param t the function argument
     * @return the function result
     */
    R apply(T t);
}

Funcation 執(zhí)行轉(zhuǎn)換操作,輸入類型 T 的數(shù)據(jù),返回 R 類型的結(jié)果,下面利用 Function 對字符串集合轉(zhuǎn)換成整型集合,并忽略掉不是數(shù)值型的字符:

private List parse(List list, Function function) {
    List result = new ArrayList<>();
    for (final String value : list) {
        // 應(yīng)用數(shù)據(jù)轉(zhuǎn)換
        if (NumberUtils.isDigits(value)) result.add(function.apply(value));
    }
    return result;
}

下面利用上面的函數(shù)式接口,將一個封裝字符串的集合轉(zhuǎn)換成整型集合,忽略不是數(shù)值形式的字符串:

FunctionDemo fd = new FunctionDemo();
List list = new ArrayList<>();
list.addAll(Arrays.asList("a", "1", "2", "3", "4", "5", "6"));
List result = fd.parse(list, (value) -> Integer.valueOf(value));
System.out.println(result);
// 輸出:[1, 2, 3, 4, 5, 6]
2.2.3 一些需要注意的事情

類型推斷

在編碼過程中,有時(shí)候可能會疑惑我們的調(diào)用代碼會具體匹配哪個函數(shù)式接口,實(shí)際上編譯器會根據(jù)參數(shù)、返回類型、異常類型(如果存在)等因素做正確的判定。在具體調(diào)用時(shí),一些時(shí)候可以省略參數(shù)的類型以進(jìn)一步簡化代碼:

// 篩選蘋果
List filterApples = filterApplesByAppleFilter(apples,
        (Apple apple) -> Color.RED.equals(apple.getColor()) && apple.getWeight() >= 100);

// 某些情況下我們甚至可以省略參數(shù)類型,編譯器會根據(jù)上下文正確判斷
List filterApples = filterApplesByAppleFilter(apples,
        apple -> Color.RED.equals(apple.getColor()) && apple.getWeight() >= 100);

局部變量

上面所有例子中使用的變量都是 lambda 表達(dá)式的主體參數(shù),我們也可以在 lambda 中使用實(shí)例變量、靜態(tài)變量,以及局部變量,如下代碼為在 lambda 表達(dá)式中使用局部變量:

int weight = 100;
List filterApples = filterApplesByAppleFilter(apples,
        apple -> Color.RED.equals(apple.getColor()) && apple.getWeight() >= weight);

上述示例我們在 lambda 中使用了局部變量 weight,不過在 lambda 中使用局部變量還是有很多限制,學(xué)習(xí)初期 IDE 可能經(jīng)常會提示我們 Variable used in lambda expression should be final or effectively final 的錯誤,即要求在 lambda 表達(dá)式中使用的變量必須 __顯式聲明為 final 或事實(shí)上的 final 類型__。

為什么要限制我們直接使用外部的局部變量呢?主要原因在于內(nèi)存模型,我們都知道實(shí)例變量在堆上分配的,而局部變量在棧上進(jìn)行分配,lambda 表達(dá)式運(yùn)行在一個獨(dú)立的線程中,了解 JVM 的同學(xué)應(yīng)該都知道棧內(nèi)存是線程私有的,所以局部變量也屬于線程私有,如果肆意的允許 lambda 表達(dá)式引用局部變量,可能會存在局部變量以及所屬的線程被回收,而 lambda 表達(dá)式所在的線程卻無從知曉,這個時(shí)候去訪問就會出現(xiàn)錯誤,之所以允許引用事實(shí)上的 final(沒有被聲明為 final,但是實(shí)際中不存在更改變量值的邏輯),是因?yàn)閷τ谠撟兞坎僮鞯氖亲兞扛北?,因?yàn)樽兞恐挡粫桓?,所以這份副本始終有效。這一限制可能會讓剛剛開始接觸函數(shù)式編程的同學(xué)不太適應(yīng),需要慢慢的轉(zhuǎn)變思維方式。

實(shí)際上在 java 8th 之前,我們在方法中使用內(nèi)部類時(shí)就已經(jīng)遇到了這樣的限制,因?yàn)樯芷诘南拗?JVM 采用復(fù)制的策略將局部變量復(fù)制一份到內(nèi)部類中,但是這樣會帶來多個線程中數(shù)據(jù)不一致的問題,于是衍生了禁止修改內(nèi)部類引用的外部局部變量這一簡單、粗暴的策略,只不過在 8th 之前必須要求這部分變量采用 final 修飾,但是 8th 開始放寬了這一限制,只要求所引用變量是 “事實(shí)上” 的 final 類型即可。

三. 方法引用

方法引用可以更近一步的簡化代碼,有時(shí)候這種簡化讓代碼看上去更加直觀,先看一個例子:

/* ... 省略apples的初始化操作 */

// 采用lambda表達(dá)式
apples.sort((Apple a, Apple b) -> Float.compare(a.getWeight(), b.getWeight()));

// 采用方法引用
apples.sort(Comparator.comparing(Apple::getWeight));

方法引用通過 :: 將方法隸屬和方法自身連接起來,主要分為三類:

靜態(tài)方法

(args) -> ClassName.staticMethod(args)
轉(zhuǎn)換成:
ClassName::staticMethod

參數(shù)的實(shí)例方法

(args) -> args.instanceMethod()
轉(zhuǎn)換成:
ClassName::instanceMethod  // ClassName是args的類型

外部的實(shí)例方法

(args) -> ext.instanceMethod(args)
轉(zhuǎn)換成:
ext::instanceMethod(args)

鑒于作者水平有限,文中不免有錯誤之處,歡迎批評指正
個人博客:www.zhenchao.org

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/70650.html

相關(guān)文章

  • Java 8th 函數(shù)編程:Optional 類型

    摘要:當(dāng)滿足條件時(shí)執(zhí)行傳入的參數(shù)化操作。最后提醒一點(diǎn),好用但不能濫用,在設(shè)計(jì)一個接口方法時(shí)是否采取類型返回需要斟酌,一味的使用會讓代碼變得比較啰嗦,反而破壞了代碼的簡潔性。鑒于作者水平有限,文中不免有錯誤之處,歡迎批評指正個人博客 NullPointException 可以說是所有 java 程序員都遇到過的一個異常,雖然 java 從設(shè)計(jì)之初就力圖讓程序員脫離指針的苦海,但是指針確實(shí)是實(shí)際...

    RayKr 評論0 收藏0
  • 深入淺出 Java 8 Lambda 表達(dá)

    摘要:在支持一類函數(shù)的語言中,表達(dá)式的類型將是函數(shù)。匿名函數(shù)的返回類型與該主體表達(dá)式一致如果表達(dá)式的主體包含一條以上語句,則表達(dá)式必須包含在花括號中形成代碼塊。注意,使用表達(dá)式的方法不止一種。 摘要:此篇文章主要介紹 Java8 Lambda 表達(dá)式產(chǎn)生的背景和用法,以及 Lambda 表達(dá)式與匿名類的不同等。本文系 OneAPM 工程師編譯整理。 Java 是一流的面向?qū)ο笳Z言,除了部分簡...

    wdzgege 評論0 收藏0
  • Java8-1-初識Lambda表達(dá)函數(shù)接口

    摘要:而在中,表達(dá)式是對象,它們必須依附于一類特別的對象類型函數(shù)式接口。即表達(dá)式返回的是函數(shù)式接口類型。 Java8被稱作Java史上變化最大的一個版本。其中包含很多重要的新特性,最核心的就是增加了Lambda表達(dá)式和Stream API。這兩者也可以結(jié)合在一起使用。首先來看下什么是Lambda表達(dá)式。Lambda表達(dá)式,維基百科上的解釋是一種用于表示匿名函數(shù)和閉包的運(yùn)算符,感覺看到這個解釋...

    jzman 評論0 收藏0
  • 企業(yè)級lambda表達(dá),讓你對lambda有更好的理解

    摘要:但,如果加入了函數(shù)式編程,也就是將方法作為形參傳遞,這必然讓開發(fā)者為難。但是,其他語言早就使用了函數(shù)式編程,比如最常見腳本語言。這就是函數(shù)式編程,傳遞的是一個函數(shù)直到,才引用了函數(shù)式編程,也就是我們所說的表達(dá)式。 導(dǎo)讀 Java從jdk1發(fā)展到今天,方法的形參類型可以是基本變量,可以是jdk自帶的類型,也可以是用戶自定義的類型,但是,方法能不能作為形參來傳遞?我們希望java能夠像其他...

    justCoding 評論0 收藏0
  • 企業(yè)級lambda表達(dá),讓你對lambda有更好的理解

    摘要:但,如果加入了函數(shù)式編程,也就是將方法作為形參傳遞,這必然讓開發(fā)者為難。但是,其他語言早就使用了函數(shù)式編程,比如最常見腳本語言。這就是函數(shù)式編程,傳遞的是一個函數(shù)直到,才引用了函數(shù)式編程,也就是我們所說的表達(dá)式。 導(dǎo)讀 Java從jdk1發(fā)展到今天,方法的形參類型可以是基本變量,可以是jdk自帶的類型,也可以是用戶自定義的類型,但是,方法能不能作為形參來傳遞?我們希望java能夠像其他...

    SegmentFault 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<