聊一聊Java中的Steam流

来源:https://www.cnblogs.com/jingdongkeji/archive/2023/07/10/17540566.html
-Advertisement-
Play Games

在我們的日常編程任務中,對於集合的製造和處理是必不可少的。當我們需要對於集合進行分組或查找的操作時,需要用迭代器對於集合進行操作,而當我們需要處理的數據量很大的時候,為了提高性能,就需要使用到並行處理,這樣的處理方式是很複雜的。流可以幫助開發者節約寶貴的時間,讓以上的事情變得輕鬆。 ...


1 引言

在我們的日常編程任務中,對於集合的製造和處理是必不可少的。當我們需要對於集合進行分組或查找的操作時,需要用迭代器對於集合進行操作,而當我們需要處理的數據量很大的時候,為了提高性能,就需要使用到並行處理,這樣的處理方式是很複雜的。流可以幫助開發者節約寶貴的時間,讓以上的事情變得輕鬆。

2 流簡介

流到底是什麼呢?簡要的定義為“從支持數據處理操作的源生成的元素序列”,接下來對於這個定義進行簡要分析。

2.1 支持數據處理操作

流的數據處理操作和資料庫的可以聲明式的指定分組或查找等功能支持類似,和函數式編程的思想一致,如filter、map、reduce、find、match、sort等操作,這些流操作可以串列執行,也可以並行執行。

2.2 源

流會使用一個提供數據的源,可以通過三種方式來創建對象流,一種是由集合對象創建流:

List<Integer> list = Arrays.asList(111,222,333);
Stream<Integer> stream = list.stream();

一種是由數組創建流:

IntStream stream = Arrays.stream(new int(){111,222,333});

一種是由靜態方法Stream.of()創建流,底層還是Arrays.stream():

Stream<Integer> stream = Stream.of(111, 222, 333);

Stream stream = Stream.of(111, 222, 333);

從有序集合生成流時會保留原有的順序。由列表生成的流,其元素順序與列表一致。
還有兩種特殊的流:

  • 空流:Stream.empty()
  • 無限流:Stream.genarate()

2.3 元素序列

流也可以和集合一樣訪問包含特定的元素類型的一組有序值,但是它們的主要目的不一樣,集合的主要目的是在於存儲和訪問元素,流的主要目的在於表達計算。

3 流的思想

流式思想和生產中的流水線具有異曲同工之妙,很多流模型都會返回一個流,這些模型都只負責它所需要做的事情,並不需要格外的記憶體空間來存儲處理的結果。這些流模型可以被鏈接起來形成一個大的流水線,我們在這個過程中不關註中間步驟的數據被如何處理,只需要使用整個流水線處理後的結果。接下來的代碼可以體現這種思想,代碼中以商品為例,我們要篩選出商品中體積大於200的前兩個商品的名字。

首先是商品類的定義:

public class Goods {
    private final String Name;
    private final Integer Volume;

    public Goods(String name, Integer volume) {
        Name = name;
        Volume = volume;
    }
    public String getName() {
        return Name;
    }
    public Integer getVolume() {
        return Volume;
    }
}

接下來是商品集合的定義:

List<Goods> goods = Arrays.asList(new Goods("土豆",10),
new Goods("冰箱",900),new Goods("辦公椅",300));

接下來獲取我們想要的結果:

List<String> twofoods = goods.stream()//獲取流
    .filter(goods1 -> goods1.getVolume()>200)//篩選商品體積大於200的
    .map(Goods::getName)//獲取商品名稱
    .limit(2)//篩選頭兩個商品
    .collect(Collectors.toList());//將結果保存在list中

這樣看來,通過流來處理我們的特定需求,是不是比使用集合的迭代要方便很多呢?

4 流處理的特性

  • 不存儲數據
  • 不會改變數據源
  • 只可被使用一次

這裡我們使用一個測試類StreamCharacteristic來驗證流處理的以上特性:

import org.springframework.util.Assert;
import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;
import java.util.stream.Stream;
public class StreamCharacteristic {
    public void test1(){
        List<Integer> list = Arrays.asList(1,2,2,5,6,9);
        list.stream().distinct();
        System.out.println(list.size());
    }
    public void test2(){
        List<String> list = Arrays.asList("wms", "KA", "5.0");
        Stream<String> stream = list.stream();
        stream.forEach(System.out::println);
        stream.forEach(System.out::println);
    }
}

test1()中的結果為6,儘管我們對於list對象所生成的Stream流做了去重操作distinct(),但是不影響數據源list。

test2()中調用了兩次 stream.forEach方法來列印每一個單詞,第二次調用時,拋出了一個“java.lang.IllegalStateException”異常:“stream has already been operated upon or closed”。這說明流不存儲數據,遍歷完後這個流已經被消費掉了,而且流不可以重覆使用。

5 流操作與流的使用

將所有的流操作連接起來可以組合成一個管道,管道有兩類操作:中間操作和終端操作。
StreamAPI常用的中間操作有:filter,map,limit,sorted,distinct。

StreamAPI常用的終端操作有:forEach,count,collect。

在使用流的時候,主要需要三個要素:一個用來執行查詢的數據源,用來形成一條流的流水線的中間操作鏈,一個能夠執行流水線並能生成結果的終端操作。

下圖展示了流的整個操作流程:

6 總結

  • 流是從支持數據處理操作的源生成的元素序列
  • 流的思想類似於生產中的流水線
  • 流不存儲數據,不改變數據源,只能被改變一次
  • 流的操作主要分為中間操作和終端操作兩大類

作者:京東物流 王辰瑋

來源:京東雲開發者社區 自猿其說Tech


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • ## JDBC 驅動載入 => 連接創建 => 創建編譯 / 預編譯語句 => 獲取結果集 => 遍歷結果集 => 返回結果集 | 介面 | | | | | | Driver | 驅動 | | Connection | 連接 | | Statement | 操作 | | ResultSet | 結果 ...
  • 引自:https://blog.csdn.net/weixin_43795921/article/details/127224633 template <typename IdentifierType, class AbstractProduct, class ProductCreator = Ab ...
  • 將bytes 轉換為long類型: 第一種方式: String 接收 bytes 的構造器轉成 String,再 Long.parseLong; 但此種情況需要註意:位元組數組中的每個位元組都必須是有效的數字字元。如果位元組數組包含非數字字元,則會引發NumberFormatException異常。確保在 ...
  • 一. 介紹 String 是Java.long包下的String類,是一個特殊的引用類型,用於表示字元串。它提供了許多方法來操作和處理字元串,比如連接、截取、查找、替換等。String類內部使用字元數組( char[] ) 來存儲字元串的內容,value欄位被final修飾,String對象一旦創建 ...
  • # 1. 配置文件 ## 1.1 外部載入順序 1. 命令行參數 java -jar spring-boot-02-config-02.0.0.1-SNAPSHOT.jar --server.port=8087 java -jar spring-boot-02-config-02.0.0.1-SNA ...
  • 電商支付實戰項目與相關面試題總結 接下來我將用一篇萬字長文,總結好這個項目以達到可以正面硬鋼面試官的水平,如果作為一個毫無頭緒的大學生的你,簡歷中需要一個還算拿得出手的項目,那麼在2023年的今天,足矣作為一個還算OK的項目寫進你的簡歷。當然,這隻能算簡歷中的第一個項目,你還需要一個更好一些的項目作 ...
  • 當聲明枚舉類型或定義一組相關常量時,Go語言中的`iota`關鍵字可以幫助我們簡化代碼並自動生成遞增的值。本文檔將詳細介紹`iota`的用法和行為。 ### `iota`關鍵字 `iota`是Go語言中的一個預定義標識符,它用於創建自增的無類型整數常量。`iota`的行為類似於一個計數器,每次在常量 ...
  • **深度探討Django ORM的概念、基礎使用、進階操作以及詳細解析在實際使用中如何處理資料庫操作。同時,我們還討論了模型深入理解,如何進行CRUD操作,並且深化理解到資料庫遷移等高級主題。為了全面解讀Django ORM,我們也討論了其存在的不足,並對其未來發展進行了展望。這篇文章旨在幫助讀者全 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...