parallelStream线程问题及解决方案

parallelStream可以在多个线程中并行处理流数据,提高性能。然而,如果在处理过程中涉及共享的可变状态,可能会导致线程不安全的问题。以下是一个简单的示例演示了如何在不正确的使用情况下导致线程安全问题:

复制代码
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;

public class ParallelStreamDemo {

    public static void main(String[] args) {
        // 使用一个普通的ArrayList来存储结果
        List<Integer> resultList = new ArrayList<>();

        // 创建一个范围从1到1000的列表
        List<Integer> numbers = new ArrayList<>();
        for (int i = 1; i <= 1000; i++) {
            numbers.add(i);
        }

        // 使用parallelStream尝试将每个数字的平方加入resultList
        numbers.parallelStream().forEach(number -> {
            // 这是一个临界区,resultList被多个线程同时修改
            resultList.add(number * number);
        });

        // 输出结果大小
        System.out.println("Expected size: " + numbers.size());
        System.out.println("Actual size: " + resultList.size());

        // 打印结果列表中的一些内容
        System.out.println("Some elements in the result: " + resultList.subList(0, 10));
    }
}

问题分析:

  1. 线程不安全ArrayList是线程不安全的。在多个线程同时进行写操作时,可能导致数据丢失或者其他不一致的问题。

  2. 结果不准确 :实际输出的resultList可能小于预期,因为当多个线程同时尝试写入同一个内存位置时,它们可能不会正确处理并发访问,导致某些写操作被忽略。

解决方案:

若要解决这样的问题,可以选择一些线程安全的集合类,例如 CopyOnWriteArrayList 或者使用同步机制来确保线程安全

复制代码
import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;

public class ParallelStreamThreadSafeDemo {

    public static void main(String[] args) {
        // 使用线程安全的CopyOnWriteArrayList来存储结果
        List<Integer> resultList = new CopyOnWriteArrayList<>();

        // 创建一个范围从1到1000的列表
        List<Integer> numbers = new ArrayList<>();
        for (int i = 1; i <= 1000; i++) {
            numbers.add(i);
        }

        // 使用parallelStream添加数字的平方到resultList
        numbers.parallelStream().forEach(number -> {
            resultList.add(number * number);
        });

        // 输出结果大小
        System.out.println("Expected size: " + numbers.size());
        System.out.println("Actual size: " + resultList.size());
        System.out.println("Some elements in the result: " + resultList.subList(0, 10));
    }
}

通过使用CopyOnWriteArrayList,我们确保了在并发写时是线程安全的,这样就可以得到预期的结果。

相关推荐
sg_knight22 分钟前
Spring Cloud与RabbitMQ深度集成:从入门到生产级实战
java·spring boot·spring·spring cloud·消息队列·rabbitmq·stream
Chan161 小时前
批处理优化:从稳定性、性能、数据一致性、健壮性、可观测性五大维度,优化批量操作
java·spring boot·后端·性能优化·java-ee·intellij-idea·优化
行者阿毅1 小时前
langchain4j+DashScope (通义千问)文生图
java·ai作画
Bug退退退1231 小时前
Java 网络流式编程
java·服务器·spring·sse
IT机器猫1 小时前
RabbitMQ
java·rabbitmq·java-rabbitmq
小杨的全栈之路1 小时前
冒泡、插入、选择、归并、堆排序:从名字由来到Java实现,一篇讲透
java·排序算法
yinke小琪1 小时前
面试官:谈谈为什么要拆分数据库?有哪些方法?
java·后端·面试
自由的疯1 小时前
java DWG文件转图片
java·后端·架构
小兔崽子去哪了1 小时前
EasyExcel 使用
java·excel
青云交1 小时前
Java 大视界 -- Java 大数据机器学习模型的对抗攻击与防御技术研究
java·机器学习模型·对抗攻击·java 大数据·防御技术·对抗训练·i - fgsm