Java stream性能比较
创始人
2025-05-31 10:20:37
0

环境

  • Ubuntu 22.04
  • IntelliJ IDEA 2022.1.3
  • JDK 17
  • CPU:8核
➜  ~ cat /proc/cpuinfo | egrep -ie 'physical id|cpu cores'
physical id	: 0
cpu cores	: 1
physical id	: 2
cpu cores	: 1
physical id	: 4
cpu cores	: 1
physical id	: 6
cpu cores	: 1
physical id	: 8
cpu cores	: 1
physical id	: 10
cpu cores	: 1
physical id	: 12
cpu cores	: 1
physical id	: 14
cpu cores	: 1

目标

文本通过实际测试,从以下几个维度比较Java stream的性能:

  • stream VS. parallelStream
  • 分步 VS. 总体,分步指的是每次操作都转换为List,下个操作前再转换为stream,而总体指的是全部操作之后再转换为List。显然,总体的性能会好于分步的性能
  • 不同数据量对性能的影响

准备

新建maven项目 test0317

打开 pom.xml 文件,添加如下内容:

        junitjunit4.13.2test

src/test/java/com.example.test0317 目录下创建package package1 ,并创建类 Test0317

package com.example.test0317.package1;import org.junit.Test;import java.util.List;
import java.util.stream.Stream;public class Test0317 {private List list1 = null;private long size = 10000000;private long start = 0;private long end = 0;private long time = 0;
}

测试

测试1(stream,10000000,分步)

    @Testpublic void test1() {System.out.println("\n****** test1: stream, " + size + ", step by step ******");for (int i = 0; i < 3; i++) {list1 = Stream.generate(Math::random).limit(size).toList();start = System.currentTimeMillis();list1 = list1.stream().map(e -> e + 1).toList();list1 = list1.stream().map(e -> e * 2).toList();list1 = list1.stream().sorted().toList();end = System.currentTimeMillis();time = end - start;System.out.println("time = " + time);}}

运行结果如下:

****** test1: stream, 10000000, step by step ******
time = 6062
time = 5931
time = 6917

测试2(parallelStream,10000000,分步)

    @Testpublic void test2() {System.out.println("\n****** test2: parallelStream, " + size + ", step by step ******");for (int i = 0; i < 3; i++) {list1 = Stream.generate(Math::random).limit(10000000).toList();start = System.currentTimeMillis();list1 = list1.parallelStream().map(e -> e + 1).toList();list1 = list1.parallelStream().map(e -> e * 2).toList();list1 = list1.parallelStream().sorted().toList();end = System.currentTimeMillis();time = end - start;System.out.println("time = " + time);}}

运行结果如下:

****** test2: parallelStream, 10000000, step by step ******
time = 2038
time = 1822
time = 2000

测试3(stream,10000000,总体)

    @Testpublic void test3() {System.out.println("\n****** test3: stream, " + size + ", whole ******");for (int i = 0; i < 3; i++) {list1 = Stream.generate(Math::random).limit(10000000).toList();start = System.currentTimeMillis();list1 = list1.stream().map(e -> e + 1).map(e -> e * 2).sorted().toList();end = System.currentTimeMillis();time = end - start;System.out.println("time = " + time);}}

运行结果如下:

****** test3: stream, 10000000, whole ******
time = 6118
time = 5774
time = 6310

测试4(parallelStream,10000000,总体)

    @Testpublic void test4() {System.out.println("\n****** test4: parallelStream, " + size + ", whole ******");for (int i = 0; i < 3; i++) {list1 = Stream.generate(Math::random).limit(10000000).toList();start = System.currentTimeMillis();list1 = list1.parallelStream().map(e -> e + 1).map(e -> e * 2).sorted().toList();end = System.currentTimeMillis();time = end - start;System.out.println("time = " + time);}}

运行结果如下:

****** test4: parallelStream, 10000000, whole ******
time = 1771
time = 1873
time = 2011

测试5(stream,20000000,分步)

运行结果如下:

****** test1: stream, 20000000, step by step ******
time = 12870
time = 12642
time = 12425

测试6(parallelStream,20000000,分步)

运行结果如下:

****** test2: parallelStream, 20000000, step by step ******
time = 4216
time = 4247
time = 4420

测试7(stream,20000000,总体)

运行结果如下:

****** test3: stream, 20000000, whole ******
time = 12199
time = 12136
time = 12088

测试8(parallelStream,20000000,总体)

运行结果如下:

****** test4: parallelStream, 20000000, whole ******
time = 3526
time = 3796
time = 4105

上面的测试中,因为CPU是8核,所以parallelStream最多使用8个线程,而下面的测试是指定使用2线程,方法为在JVM的启动选项(VM options)里设置 -Djava.util.concurrent.ForkJoinPool.common.parallelism=2 ,如下图所示:

在这里插入图片描述

测试9(2线程,parallelStream,10000000,分步)

运行结果如下:

****** test2: parallelStream, 10000000, step by step ******
time = 3446
time = 3246
time = 3523

测试10(2线程,parallelStream,10000000,总体)

运行结果如下:

****** test4: parallelStream, 10000000, whole ******
time = 3173
time = 3136
time = 3259

测试11(2线程,parallelStream,20000000,分步)

运行结果如下:

****** test2: parallelStream, 20000000, step by step ******
time = 7246
time = 7830
time = 7613

测试12(2线程,parallelStream,20000000,总体)

运行结果如下:

****** test4: parallelStream, 20000000, whole ******
time = 7292
time = 7438
time = 7109

总结

测试结果总结如下:

stream VS. parallelStreamstepwise VS. whole元素个数平均时间(秒)速度提升
测试1streamstepwise100000006.3baseline
测试2parallelStreamstepwise100000002.03.15
测试3streamwhole100000006.11.03
测试4parallelStreamwhole100000001.93.32

总结:在8核,10000000个元素的情况下,parallelStream相比stream性能提升很大,而总体相比分步只是略有性能提升。

如果把10000000个元素换为20000000个元素,测试结果如下:

stream VS. parallelStreamstepwise VS. whole元素个数平均时间(秒)速度提升
测试5streamstepwise2000000012.6baseline
测试6parallelStreamstepwise200000004.32.93
测试7streamwhole2000000012.11.04
测试8parallelStreamwhole200000003.83.32

可见,如果元素个数加倍,则对于每个测试结果,运行时间也都几乎加倍,符合线性增长。

总结:在8核,20000000个元素的情况下,parallelStream相比stream性能提升很大,而总体相比分步只是略有性能提升。

另外,若换成2线程,其性能显然在单线程和8线程之间。测试结果如下:

stream VS. parallelStreamstepwise VS. whole元素个数平均时间(秒)速度提升
测试9parallelStreamstepwise100000003.31.91
测试10parallelStreamwhole100000003.12.03
测试11parallelStreamstepwise200000007.61.66
测试12parallelStreamwhole200000007.31.73

可见,2线程相比单线程,性能提升接近于2倍,但是达不到2倍,这是因为创建和切换线程需要消耗一定的时间和资源,同理,拆分及合并数据也需要消耗一定的时间和资源。

总结:在2线程,10000000或20000000个元素的情况下,parallelStream相比stream的性能提升接近于2倍,而总体相比分步只是略有性能提升。

最后多说一句:在数据量很大(本例中达到千万级别)时,parallelStream相比stream而言,性能有非常大的提升。但是若数据量不大,比如我测试了10000,则parallelStream相比stream,性能不但没有提升,甚至变得更差了,原因前面已经提到了。

不过话说回来,即使parallelStream比起stream性能变差,但因为数据量小,所以消耗的时间总量就少,比如说假设从10毫秒变成15毫秒,虽然多了50%的时间消耗,但是因为绝对值很小,所以问题不大。

从这个角度看来,还是应该尽量用parallelStream来取代stream。

当然,本例只是一个非常简单的模型,在一些复杂的情况下,比如有线程安全的问题,就要考虑应该用stream还是parallelStream。

相关内容

热门资讯

兴银研究精选股票A:2025年... AI基金兴银研究精选股票A(008537)披露2025年四季报,第四季度基金利润398.92万元,加...
2026全球服务商大会:为中国... 转自:新华财经 新华财经上海1月25日电(记者 郭敬丹、有之炘)自2019年首次发布以来,上海市静安...
权威访谈 开局“十五五”丨央行... “十五五”开局之年,适度宽松的货币政策如何发力?总台央视记者对中国人民银行行长潘功胜进行了专访。潘功...
「图解牛熊股」贵金属概念涨幅居... 财联社1月25日讯,本周A股三大指数涨跌不一,其中上证指数周涨0.83%,深成指周涨1.11%,创业...
同方股份原总裁、董事长陆致成因... 据微信公众号“同方股份有限公司”1月24日晚消息,同方股份有限公司原总裁、董事长陆致成,因病于202...
欧洲股市结束五周连涨 航空股走... 来源:环球市场播报 欧洲股市遭遇两个月来最大单周跌幅,地缘政治风险持续,油价飙升拖累航空股走低。 斯...
湖南外贸优品进机关,九芝堂等8... 华声在线1月23日讯(通讯员 刘海江)今天,“湖南外贸优品进机关”专场活动在省人大常委会院内举办,九...
财政金融协同,打出支持民企“组... 面向“十五五”,要持续加强民营企业金融服务,努力做到金融对民营经济的支持与民营经济对经济社会发展的贡...
【数据发布】2025年1-12... 2025年1-12月份 全市社会消费品零售总额 运行情况 四季度,全市实现社会消费品零售总额281....
华南第一,名匠摇篮:鼎才CNC... 在深圳龙华一处价值数千万的模拟工厂里,一位学员正通过AR眼镜调试德玛吉五轴机床的程序,隔壁就业屏上,...
黎瑞刚出手!邵氏兄弟+正午阳光... 资深传媒人黎瑞刚又有大动作。 日前,港股邵氏兄弟控股(简称“邵氏兄弟”)披露了一则“蛇吞象”式收购计...
对招商银行、中信银行、浦发银行... 文 | CFN 大河 2025年上市银行首批业绩快报陆续出炉,招商银行、中信银行、浦发银行、兴业银行...
丽人丽妆转型阵痛持续:亏损扩大... 本报(chinatimes.net.cn)记者方凤娇 上海报道 国内美妆代运营头部企业丽人丽妆(60...
“平头哥”最新动作!深度拆解阿... 上海浦东新区张江人工智能产业园内,一座灰橙交融的建筑静静伫立,平头哥半导体有限公司(以下简称平头哥)...
鲁泰纺织股份有限公司 2025... 证券代码:000726 200726 证券简称:鲁泰A 鲁泰B 公告编号:2026-002 债券代码...
特朗普再发关税威胁!加拿大回应 【导读】特朗普威胁对加拿大征收100%关税,加总理回应 中国基金报记者 李智 一起来关注下海外资讯。...
被特朗普暴击的马克龙,又放弃威... 被特朗普暴击的马克龙,现在又放弃威胁中方了,突然变脸希望中国投资欧洲,表明西方国家已经开始觉醒。 最...
原创 异... 在浩瀚的星空下,每一颗星星都承载着一个关于爱情的故事。异地恋,这个让无数情侣夜不能寐的话题,究竟能否...
小儿癫痫认知康复护理实践 小儿癫痫认知康复护理方法,涵盖康复认知护理、家庭认知康复护理辅助、护理宣教等手段,以有效促进患儿康复...
钻戒回收价腰斩,银保温杯身价翻... 谁能想到,贵金属市场正在上演一出荒诞大戏 就在上个月,朋友小琳还满心欢喜地戴着新买的钻戒。那是她存...