0

0

Java中多线程并发计算数组最大值的正确同步实践

心靈之曲

心靈之曲

发布时间:2026-02-19 08:23:13

|

881人浏览过

|

来源于php中文网

原创

Java中多线程并发计算数组最大值的正确同步实践

本文详解如何在多线程分治求最大值场景中,通过合理使用join()和线程安全设计(而非错误地同步run()方法)确保主线程正确等待子线程完成并安全获取结果。

本文详解如何在多线程分治求最大值场景中,通过合理使用join()和线程安全设计(而非错误地同步run()方法)确保主线程正确等待子线程完成并安全获取结果。

在Java并发编程中,一个常见误区是:当主线程需要汇总多个子线程的计算结果时,试图通过synchronized修饰run()或getMax()方法来“解决竞态”,但这不仅逻辑错误,更会严重损害并发性。问题本质并非数据竞争(max字段本身无跨线程读写冲突),而是线程执行顺序未受控导致的时序错误——主线程在子线程尚未完成run()前就调用了getMax(),读取到未初始化的默认值0。

✅ 正确解法:用 join() 显式等待线程终止

Thread.join() 是专为解决此类“主线程依赖子线程完成”问题而设计的标准机制。它阻塞当前线程,直到目标线程自然结束,无需加锁,语义清晰且高效。

以下是修复后的完整示例(关键修改已高亮):

NewsBang
NewsBang

盛大旗下AI团队推出的智能新闻阅读App

下载
public class MaxTask extends Thread {
    private final int[] arr;  // 建议final,避免意外修改
    private int max;
    private final int first, last;

    public MaxTask(int[] arr, int first, int last) {
        this.arr = arr;
        this.first = first;
        this.last = last;
    }

    public int getMax() {
        return max; // 无需synchronized:仅在join后调用,此时max已稳定
    }

    @Override
    public void run() { // 不应synchronized!否则所有任务串行执行,失去多线程意义
        if (first > last || arr.length == 0) return;
        max = arr[first];
        for (int i = first + 1; i <= last; i++) {
            if (arr[i] > max) max = arr[i];
        }
    }
}
public class MainMax {
    public static void main(String[] args) throws Exception {
        int size = 100;
        int workers = 10;
        int[] arr = new int[size];
        // 初始化数组...
        for (int i = 0; i < size; i++) {
            arr[i] = (int)(Math.random() * 100);
        }

        int gsize = (arr.length - 1) / workers;
        MaxTask[] tasks = new MaxTask[workers];

        // 启动所有子线程
        int first = 0;
        for (int i = 0; i < workers; i++) {
            int last = Math.min(first + gsize, arr.length - 1); // 修正边界:避免越界
            tasks[i] = new MaxTask(arr, first, last);
            tasks[i].start();
            first = last + 1;
        }

        // ✅ 关键步骤:主线程等待所有子线程完成
        for (MaxTask task : tasks) {
            task.join(); // 阻塞直到task.run()执行完毕
        }

        // 此时所有max均已计算完成,可安全读取
        int maxmax = tasks[0].getMax();
        for (int i = 1; i < tasks.length; i++) {
            int temp = tasks[i].getMax();
            if (temp > maxmax) maxmax = temp;
        }
        System.out.println("maxmax=" + maxmax);
    }
}

⚠️ 重要注意事项

  • 不要同步 run() 方法:synchronized void run() 会使所有 MaxTask 实例串行执行,彻底抵消多线程优势,性能可能比单线程更差。
  • getMax() 无需同步:因 join() 保证了调用 getMax() 时 run() 已结束,max 字段状态已确定,不存在可见性问题(join() 具有内存屏障语义,确保之前写入对后续读取可见)。
  • 边界处理要严谨:原代码中 last = first + gsize 可能导致索引越界(如 arr.length=100, workers=10 → gsize=9, 最后一组 last=99 合理;但若 size=101 则需 Math.min(...) 防御)。
  • 现代替代方案推荐:生产环境建议使用 ExecutorService + Future 或 ForkJoinPool,它们提供更健壮的生命周期管理、异常处理和资源复用能力。

总结

解决“主线程过早读取子线程结果”的核心在于控制执行时序,而非引入不必要的同步。Thread.join() 是轻量、标准且语义明确的解决方案。理解 join() 的阻塞行为与内存可见性保证,是编写可靠多线程程序的基础。盲目添加 synchronized 不仅不能解决问题,反而会掩盖真实缺陷并引入性能瓶颈。

立即学习Java免费学习笔记(深入)”;

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

183

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

115

2025.11.27

length函数用法
length函数用法

length函数用于返回指定字符串的字符数或字节数。可以用于计算字符串的长度,以便在查询和处理字符串数据时进行操作和判断。 需要注意的是length函数计算的是字符串的字符数,而不是字节数。对于多字节字符集,一个字符可能由多个字节组成。因此,length函数在计算字符串长度时会将多字节字符作为一个字符来计算。更多关于length函数的用法,大家可以阅读本专题下面的文章。

951

2023.09.19

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

675

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

366

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

本专题整合了java多线程相关教程,阅读专题下面的文章了解更多详细内容。

26

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

24

2026.01.21

C# 多线程与异步编程
C# 多线程与异步编程

本专题深入讲解 C# 中多线程与异步编程的核心概念与实战技巧,包括线程池管理、Task 类的使用、async/await 异步编程模式、并发控制与线程同步、死锁与竞态条件的解决方案。通过实际项目,帮助开发者掌握 如何在 C# 中构建高并发、低延迟的异步系统,提升应用性能和响应速度。

98

2026.02.06

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

561

2026.02.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3.7万人学习

C# 教程
C# 教程

共94课时 | 9.7万人学习

Java 教程
Java 教程

共578课时 | 67.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号