JUC工具类: CyclicBarrier详解

简介: `CyclicBarrier`是并发编程领域一个非常实用的同步辅助类,适用于并行任务场景,它提供了一种简便的线程同步机制。正确地理解和使用这个工具,对开发者来说,可以大大简化并行处理逻辑的复杂度,增强代码的健壮性与可维护性。

在并发编程的领域,Java的 CyclicBarrier是一个同步辅助类,用于管理一组线程之间的合作流程,确保这些线程在继续执行之前达到一个公共的屏障点。这是一种非常有用的工具,特别适合于处理并行任务,在所有相关任务必须互相等待对方到达某一个点后才能继续执行的情况。

CyclicBarrier,顾名思义,循环使用的屏障。它允许一组线程互相等待,直到所有线程都达到了共同的屏障点(Barrier Point),在此之后,这些线程可以选择继续执行或执行某种公共的动作。与之类似的同步辅助类是 CountDownLatch,然而不同的是,CyclicBarrier可以在达到屏障时重置,所以它可以被重新使用。

特点和用法

  1. 屏障触发执行CyclicBarrier支持一个可选的 Runnable命令,在屏障点上,当给定数量的参与线程都到达了后,该命令只会被执行一次,常被用作聚合任务或更新共享状态。
  2. 线程参与数量:在构造 CyclicBarrier的时候,你需要指定等待的线程数量。只有当足够数量的线程都执行了 await()调用时,屏障才会打开,线程才能继续执行。
  3. 循环使用:与 CountDownLatch一次性的特性不同,CyclicBarrier可以通过其 reset()方法重置屏障,以便同一组线程可以多次使用它。
  4. 超时与中断CyclicBarrierawait()方法支持超时参数和可以响应中断。如果任一线程 await()期间被中断,或者等待超时,所有正在等待的线程都将通过 BrokenBarrierException异常来响应这种情况。

实践场景

常见的使用案例包括并行计算逻辑,其中多个子任务在继续执行下一步之前必须等待对方完成。例如,在多个玩家的游戏中,可能需要等待所有玩家都做好准备,或者在科学计算中,涉及多步骤的计算,每一步都依赖于前一步的结果。

示例代码

import java.util.concurrent.CyclicBarrier;

public class CyclicBarrierExample {
    // 在这里,假定屏障打开之前,需要有3个线程到达。
    private static final int PARTIES = 3;
    private static final CyclicBarrier barrier = new CyclicBarrier(PARTIES, 
        new Runnable() {
            // 当所有线程到达屏障时,首先执行的动作。
            public void run() {
                System.out.println("所有线程已到达屏障点,屏障现在开放,准备执行接下来的动作。");
            }
        }
    );

    public static void main(String[] args) {
        for (int i = 0; i < PARTIES; i++) {
            new Thread(new Worker(i)).start();
        }
    }

    static class Worker implements Runnable {
        private final int threadNumber;

        Worker(int threadNumber) {
            this.threadNumber = threadNumber;
        }

        public void run() {
            System.out.println("线程 " + threadNumber + " 正在工作。");
            try {
                // 该线程完成工作后,等待其他线程到达屏障点
                CyclicBarrierExample.barrier.await();
            } catch (Exception e) {
                e.printStackTrace();
            }
            System.out.println("线程 " + threadNumber + " 完成了等待,继续执行它的任务。");
        }
    }
}

在这段代码中,我们定义了一个 Worker内部类,模拟的是到达屏障点的线程执行体。一个 CyclicBarrier实例被创建,设置了必须有3个线程达到屏障点之后才能触发屏障打开的机制,并在屏障触发时输出相应信息。每个线程在到达屏障点后,等待 barrier.await()直到其他线程也到达,然后所有线程会几乎同时继续执行。

结论

CyclicBarrier是并发编程领域一个非常实用的同步辅助类,适用于并行任务场景,它提供了一种简便的线程同步机制。正确地理解和使用这个工具,对开发者来说,可以大大简化并行处理逻辑的复杂度,增强代码的健壮性与可维护性。

目录
相关文章
|
Java 数据库连接 开发者
Java的Shutdown Hook机制:优雅地关闭应用程序
Java的Shutdown Hook机制:优雅地关闭应用程序
693 1
|
负载均衡
《从零开始手写RPC框架》电子书升级啦!
大家好,我是冰河~~ 今天跟大家正式宣布一个好消息,冰河的《从零开始手写RPC框架(基础篇)》电子书,升级啦! 这本电子书中包含哪些内容呢?小伙伴们可以直接拉到文末查看获取方式,也可以先听冰河吹一吹这本电子书的内容。
635 0
《从零开始手写RPC框架》电子书升级啦!
|
存储 安全 算法
3.【Elasticsearch】Elasticsearch从入门到放弃-权重及打分
【Elasticsearch】Elasticsearch从入门到放弃-权重及打分
3.【Elasticsearch】Elasticsearch从入门到放弃-权重及打分
|
存储 关系型数据库 MySQL
"深入探索MySQL临时表:性能优化利器,数据处理的灵活之选"
【8月更文挑战第9天】MySQL临时表专为存储临时数据设计,自动创建与删除,仅在当前会话中存在,有助于性能优化。它分为本地临时表和全局临时表(通过特定逻辑模拟)。创建语法类似于普通表,但加TEMPORARY或TEMP关键字。适用于性能优化、数据预处理和复杂查询,需注意内存占用和事务支持问题。合理使用可大幅提升查询效率。
771 2
|
JSON 安全 Java
解决Java中跨域请求问题的方法
解决Java中跨域请求问题的方法
|
设计模式 程序员 领域建模
还原真实世界,领域模型教你让系统自然生长
本文介绍了领域模型在软件开发中的重要性,强调了领域驱动设计(DDD)的核心在于理解业务、抽象业务,而非单纯掌握技术模式。通过对比贫血模型和充血模型,阐述了领域建模的目标是让设计随需求自然成长,保持系统的灵活性和稳定性。文章鼓励开发者在需求变更时,回归真实世界寻找解决方案,确保模型贴近业务本质。
196 4
|
Prometheus 监控 数据可视化
Grafana 插件生态系统:扩展你的监控能力
【8月更文第29天】Grafana 是一个流行的开源平台,用于创建和共享统计数据的仪表板和可视化。除了内置的支持,Grafana 还有一个强大的插件生态系统,允许用户通过安装插件来扩展其功能。本文将介绍一些 Grafana 社区提供的插件,并探讨它们如何增强仪表盘的功能性。
982 3
|
机器学习/深度学习 分布式计算 并行计算
性能优化视角:Python与R在大数据与高性能机器学习中的选择
【8月更文第6天】随着数据量的激增,传统的单机计算已经难以满足处理大规模数据集的需求。Python和R作为流行的数据科学语言,各自拥有独特的特性和生态系统来应对大数据和高性能计算的挑战。本文将从性能优化的角度出发,探讨这两种语言在处理大数据集和高性能计算时的不同表现,并提供具体的代码示例。
337 3
|
机器学习/深度学习 算法 vr&ar
Theta方法:一种时间序列分解与预测的简化方法
Theta方法整合了两个基本概念:分解时间序列和利用基本预测技术来估计未来的价值。
530 0
|
负载均衡 Java 应用服务中间件
Ribbon、Feign和OpenFeign的区别来了
Ribbon、Feign和OpenFeign的区别来了
947 2