亚洲在线久爱草,狠狠天天香蕉网,天天搞日日干久草,伊人亚洲日本欧美

為了賬號安全,請及時綁定郵箱和手機立即綁定
已解決430363個問題,去搜搜看,總會有你想問的

BiFunction - apply 方法應該總是返回第二個參數。為什么?

BiFunction - apply 方法應該總是返回第二個參數。為什么?

子衿沉夜 2023-05-10 13:49:10
我正在嘗試從其前身大于其價值的列表中識別數字。在 lambda 表達式中,如果我返回 b,它的行為符合預期,但如果我返回 a,它會給出錯誤的輸出。這兩個 return 語句有什么區別?    List<Integer> list = Arrays.asList(1,2,3,4,5,8,7);    List<Integer> result = new ArrayList<>();    list.stream().reduce((a,b) -> {if (a < b) result.add(a);return b;});    System.out.println(result);    result.clear();    list.stream().reduce((a,b) -> {if (a < b) result.add(a);return a;});    System.out.println(result);輸出:[1, 2, 3, 4, 5][1, 1, 1, 1, 1, 1]
查看完整描述

2 回答

?
白衣非少年

TA貢獻1155條經驗 獲得超0個贊

您正在執行的減少 - 使用Optional<T> java.util.stream.Stream.reduce(BinaryOperator<T> accumulator)- 等效于以下偽代碼(取自 Javadoc):


boolean foundAny = false;

T result = null;

for (T element : this stream) {

    if (!foundAny) {

        foundAny = true;

        result = element;

    } else {

        result = accumulator.apply(result, element);

    }

}

return foundAny ? Optional.of(result) : Optional.empty();

如您所見,歸約result的中間值作為累加器的第一個參數傳遞,當前element的Stream作為第二個參數傳遞。


因此,當您的 lambda 表達式返回第二個參數b(第一個片段)時,中間結果成為 的當前元素,您將在下一次迭代中將Stream其添加到。List


當您的 lambda 返回第一個參數a(第二個片段)時,中間結果保持不變(始終1是 的第一個元素Stream),并且您繼續將該值添加到List.


讓我們用實際數字來驗證一下:


result被初始化為 Stream 的第一個元素1。


然后,第一個片段調用accumulator.apply(1,2),添加1到List并返回2(這成為新的中間結果)。下一次迭代將添加2到List并返回3。等等...


第二個代碼片段調用accumulator.apply(1,2),添加1到List并返回 1(這仍然是新的中間結果)。下一次迭代將再次添加1并List再次返回1。等等...


總而言之,您的兩個累積函數具有不同的功能:


第一個結果是的最后一個元素Stream(因為它不斷丟棄當前結果并將其替換為當前元素)。


第二個結果是的第一個元素Stream(因為它保留第一個元素并忽略所有其他元素)。


查看完整回答
反對 回復 2023-05-10
?
動漫人物

TA貢獻1815條經驗 獲得超10個贊

這不是您問題的直接答案,但由于我說有狀態過濾器會更好,所以我覺得有必要展示它。采用以下有狀態謂詞:


public class GreaterThanPreceding implements Predicate<Integer> {

    private Integer preceding = null;


    @Override

    public boolean test(Integer current) {

        boolean greaterThan = preceding == null || (current > preceding);

        preceding = current;

        return greaterThan;

    }

}

這樣,流操作將如下所示:


List<Integer> collected = list.stream()

        .filter(new GreaterThanPreceding())

        .collect(Collectors.toList());

System.out.println(collected);


查看完整回答
反對 回復 2023-05-10
  • 2 回答
  • 0 關注
  • 123 瀏覽
慕課專欄
更多

添加回答

舉報

0/150
提交
取消
微信客服

購課補貼
聯系客服咨詢優惠詳情

幫助反饋 APP下載

慕課網APP
您的移動學習伙伴

公眾號

掃描二維碼
關注慕課網微信公眾號