91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

stream中怎么正確使用peek

發布時間:2023-05-11 16:26:15 來源:億速云 閱讀:245 作者:iii 欄目:開發技術

這篇文章主要講解了“stream中怎么正確使用peek”,文中的講解內容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“stream中怎么正確使用peek”吧!

簡介

自從JDK中引入了stream之后,仿佛一切都變得很簡單,根據stream提供的各種方法,如map,peek,flatmap等等,讓我們的編程變得更美好。

事實上,我也經常在項目中看到有些小伙伴會經常使用peek來進行一些業務邏輯處理。

peek的定義和基本使用

先來看看peek的定義:

    Stream<T> peek(Consumer<? super T> action);

peek方法接受一個Consumer參數,返回一個Stream結果。

而Consumer是一個FunctionalInterface,它需要實現的方法是下面這個:

    void accept(T t);

accept對傳入的參數T進行處理,但是并不返回任何結果。

我們先來看下peek的基本使用:

    public static void peekOne(){
        Stream.of(1, 2, 3)
                .peek(e -> log.info(String.valueOf(e)))
                .toList();
    }

運行上面的代碼,我們可以得到:

[main] INFO com.flydean.Main - 1
[main] INFO com.flydean.Main - 2
[main] INFO com.flydean.Main - 3

邏輯很簡單,就是打印出Stream中的元素而已。

peek的流式處理

peek作為stream的一個方法,當然是流式處理的。接下來我們用一個具體的例子來說明流式處理具體是如何操作的。

    public static void peekForEach(){
        Stream.of(1, 2, 3)
                .peek(e -> log.info(String.valueOf(e)))
                .forEach(e->log.info("forEach"+e));
    }

這一次我們把toList方法替換成了forEach,通過具體的打印日志來看看到底發生了什么。

[main] INFO com.flydean.Main - 1
[main] INFO com.flydean.Main - forEach 1
[main] INFO com.flydean.Main - 2
[main] INFO com.flydean.Main - forEach 2
[main] INFO com.flydean.Main - 3
[main] INFO com.flydean.Main - forEach 3

通過日志,我們可以看出,流式處理的流程是對應流中的每一個元素,分別經歷了peek和forEach操作。而不是先把所有的元素都peek過后再進行forEach。

Stream的懶執行策略

之所有會有流式操作,就是因為可能要處理的數據比較多,無法一次性加載到內存中。

所以為了優化stream的鏈式調用的效率,stream提供了一個懶加載的策略。

什么是懶加載呢?

就是說stream的方法中,除了部分terminal operation之外,其他的都是intermediate operation.

比如count,toList這些就是terminal operation。當接受到這些方法的時候,整個stream鏈條就要執行了。

而peek和map這些操作就是intermediate operation。

intermediate operation的特點是立即返回,如果最后沒有以terminal operation結束,intermediate operation實際上是不會執行的。

我們來看個具體的例子:

    public static void peekLazy(){
        Stream.of(1, 2, 3)
                .peek(e -> log.info(String.valueOf(e)));
    }

運行之后你會發現,什么輸出都沒有。

這表示peek中的邏輯并沒有被調用,所以這種情況大家一定要注意。

peek為什么只被推薦在debug中使用

如果你閱讀過peek的文檔,你可能會發現peek是只被推薦在debug中使用的,為什么呢?

JDK中的原話是這樣說的:

In cases where the stream implementation is able to optimize away the production of some or all the elements (such as with short-circuiting operations like findFirst, or in the example described in count), the action will not be invoked for those elements.

翻譯過來的意思就是,因為stream的不同實現對實現方式進行了優化,所以不能夠保證peek中的邏輯一定會被調用。

我們再來舉個例子:

    public static void peekNotExecute(){
        Stream.of(1, 2, 3)
                .peek(e -> log.info("peekNotExecute"+e))
                .count();
    }

這里的terminal operation是count,表示對stream中的元素進行統計。

因為peek方法中參數是一個Consumer,它不會對stream中元素的個數產生影響,所以最后的運行結果就是3。

peek中的日志輸出并沒有打印出來,表示peek沒有被執行。

所以,我們在使用peek的時候,一定要注意peek方法是否會被優化。要不然就會成為一個隱藏很深的bug。

peek和map的區別

好了,講到這里,大家應該對peek有了一個全面的認識了。但是stream中還有一個和peek類似的方法叫做map。他們有什么區別呢?

前面我們講到了peek方法需要的參數是Consumer,而map方法需要的參數是一個Function:

    <R> Stream<R> map(Function<? super T, ? extends R> mapper);

Function也是一個FunctionalInterface,這個接口需要實現下面的方法:

    R apply(T t);

可以看出apply方法實際上是有返回值的,這跟Consumer是不同的。所以一般來說map是用來修改stream中具體元素的。 而peek則沒有這個功能。

peek方法接收一個Consumer的入參. 了解&lambda;表達式的應該明白 Consumer的實現類應該只有一個方法,該方法返回類型為void. 它只是對Stream中的元素進行某些操作,但是操作之后的數據并不返回到Stream中,所以Stream中的元素還是原來的元素.

map方法接收一個Function作為入參. Function是有返回值的, 這就表示map對Stream中的元素的操作結果都會返回到Stream中去.

  • 要注意的是,peek對一個對象進行操作的時候,雖然對象不變,但是可以改變對象里面的值。

大家可以運行下面的例子:

    public static void peekUnModified(){
        Stream.of(1, 2, 3)
                .peek(e -> e=e+1)
                .forEach(e->log.info("peek unModified"+e));
    }
    public static void mapModified(){
        Stream.of(1, 2, 3)
                .map(e -> e=e+1)
                .forEach(e->log.info("map modified"+e));
    }

感謝各位的閱讀,以上就是“stream中怎么正確使用peek”的內容了,經過本文的學習后,相信大家對stream中怎么正確使用peek這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

永登县| 永康市| 安平县| 隆林| 汝城县| 梅州市| 桐乡市| 灌阳县| 彭山县| 确山县| 湘乡市| 东山县| 淅川县| 龙海市| 政和县| 兰溪市| 平乡县| 盈江县| 新宁县| 手游| 卓尼县| 安顺市| 宁津县| 大同县| 安康市| 宁国市| 新晃| 吉安县| 开江县| 沙田区| 安吉县| 广昌县| 益阳市| 萨嘎县| 广河县| 万荣县| 临海市| 广宗县| 财经| 南投县| 黔江区|