并行流parallelStream.map().collect()

一、使用场景

  1. 先贴代码
java 复制代码
public static void main(String[] args) {
    List<String> stringList = new ArrayList<>();
    List<Integer> integerList = new ArrayList<>();

    int num = 10000;
    for (int i = 0;i<num;i++){
        stringList.add(String.valueOf(i));
    }

    stringList.parallelStream().forEach(str->{
        integerList.add(Integer.parseInt(str));
    });

    System.out.println(stringList.size());
    System.out.println(integerList.size());
}

--- 下面是执行一次的结果 ---
10000
5769

可以看出来:parallelStream实际上是为每一个流创建了一个线程,去执行后面的任务。由于是多线程环境下,流后面的foreach、map里的逻辑如果有共享变量需要自行保证线程安全。这里就是因为ArrayList不是线程安全的,所以integerList与stringList的数据不一致。

  1. 结论:用于多线程执行任务

二、进阶使用

上面的代码线程不安全,如何修改?

方式一,将integerList的实现改为线程安全的List

java 复制代码
public static void main(String[] args) {
    List<String> stringList = new ArrayList<>();
    List<Integer> integerList = Collections.synchronizedList(new ArrayList<>());

    int num = 10000;
    for (int i = 0;i<num;i++){
        stringList.add(String.valueOf(i));
    }

    stringList.parallelStream().forEach(str->{
        integerList.add(Integer.parseInt(str));
    });

    System.out.println(stringList.size());
    System.out.println(integerList.size());
}

--- 下面是执行一次的结果 ---
10000
10000

方式二:使用collect

parallelStream的map,collect,flatMap,reduce等操作已经自行保证了线程安全

java 复制代码
public static void main(String[] args) {
    List<String> stringList = new ArrayList<>();

    int num = 10000;
    for (int i = 0; i < num; i++) {
        stringList.add(String.valueOf(i));
    }

    List<Integer> integerList = stringList
            .parallelStream()
            .map(Integer::parseInt)
            .collect(Collectors.toList());

    System.out.println(stringList.size());
    System.out.println(integerList.size());
}

三、常用案例

有时候,我们会for循环调用service获取多个List<T>的,最后把所有获取到的List<T>汇总起来,添加到结果集里。

  1. 用List<callable<List<T>>> callableList将多个并发任务添加进去,然后使用List<Future> futures = threadPoolExecutor.invokeAll(callableList),最后将所有future的调用结果get出来,添加到result
  2. 用parallelStream().map.collect
java 复制代码
List<User> resultList = paramList
                .parallelStream()
                // 执行业务逻辑,根据参数查询n次
                .map(param -> {
                    //todo 构建参数
                    return userService.query(param);
                })
                .collect(Collectors.toList());
相关推荐
知识即是力量ol4 分钟前
初识 Kafka(一):分布式流平台的定义、核心优势与架构全景
java·分布式·kafka·消息队列
爱吃生蚝的于勒8 分钟前
【Linux】线程概念(一)
java·linux·运维·服务器·开发语言·数据结构·vim
kong790692810 分钟前
Nginx性能优化
java·nginx·性能优化
Pluchon11 分钟前
硅基计划4.0 算法 简单模拟实现位图&布隆过滤器
java·大数据·开发语言·数据结构·算法·哈希算法
我命由我1234511 分钟前
Java 泛型 - Java 泛型通配符(上界通配符、下界通配符、无界通配符、PECS 原则)
java·开发语言·后端·java-ee·intellij-idea·idea·intellij idea
Seven9712 分钟前
AQS深度探索:以ReentrantLock看Java并发编程的高效实现
java
4311媒体网20 分钟前
C语言操作符全解析 C语言操作符详解
java·c语言·jvm
淡忘_cx20 分钟前
使用Jenkins自动化部署spring-java项目+宝塔重启项目命令(2.528.2版本)
java·自动化·jenkins
毕设源码-钟学长25 分钟前
【开题答辩全过程】以 基于SSM的孤儿救助信息管理系统设计与实现为例,包含答辩的问题和答案
java
独自破碎E25 分钟前
【曼哈顿距离】BISHI25 最大 FST 距离
java·开发语言