Java中的CyclicBarrier使用方式

CyclicBarrier是Java并发包java.util.concurrent中提供的一个同步辅助类,它允许一组线程相互等待,直到到达某个公共屏障点(common barrier point)后再继续执行。CyclicBarrier之所以被称为“循环”的,是因为一旦所有参与线程到达了屏障点,这个屏障就可以被重用。

CyclicBarrier的基本原理:

  1. 构造
    • CyclicBarrier(int parties):创建一个CyclicBarrier实例,参数parties定义了参与屏障同步的线程数。
    • CyclicBarrier(int parties, Runnable barrierAction):除了定义线程数外,还可以传入一个Runnable,当所有线程到达屏障时,这个Runnable会被其中一个线程执行。
  1. 使用
    • 每个线程在到达某个点时调用await()方法,这会使线程进入等待状态,直到所有参与线程都调用了await()
    • 当所有线程都到达屏障点后,它们会被释放并继续执行。
    • 如果某个线程在等待过程中被中断,或者有线程抛出了BrokenBarrierExceptionTimeoutException,则整个屏障被破坏,所有等待的线程将被释放并抛出异常。

CyclicBarrier的使用示例:

下面是一个使用CyclicBarrier的示例代码,模拟了一个场景,其中多个线程需要同时启动一个计算任务,只有当所有线程都准备好了,计算才会开始。

import java.util.concurrent.BrokenBarrierException;
import java.util.concurrent.CyclicBarrier;

public class CyclicBarrierExample {

    public static void main(String[] args) {
        final int numberOfThreads = 5;
        CyclicBarrier barrier = new CyclicBarrier(numberOfThreads, () -> {
            System.out.println("所有线程已准备好,开始计算...");
        });

        for (int i = 0; i < numberOfThreads; i++) {
            new Thread(() -> {
                try {
                    System.out.println(Thread.currentThread().getName() + " 正在准备...");
                    Thread.sleep((long) (Math.random() * 1000)); // 模拟准备时间
                    System.out.println(Thread.currentThread().getName() + " 准备完毕,等待其他线程...");
                    barrier.await(); // 等待所有线程到达屏障
                    System.out.println(Thread.currentThread().getName() + " 继续执行...");
                } catch (InterruptedException | BrokenBarrierException e) {
                    Thread.currentThread().interrupt();
                    System.err.println("线程中断:" + e.getMessage());
                }
            }).start();
        }
    }
}

在这个例子中,我们创建了一个CyclicBarrier实例,指定有5个线程参与。当所有5个线程都调用了await()方法时,屏障被触发,此时传递给CyclicBarrier构造函数的Runnable被调用,打印出所有线程已准备好的消息,随后所有线程继续执行。

注意事项:

  • 如果线程在等待过程中被中断,那么它会抛出InterruptedException
  • 如果一个线程在await()方法中抛出了异常(除了InterruptedException),那么屏障将被破坏,其他所有等待的线程将会接收到BrokenBarrierException
  • CyclicBarrier可以重用,当所有线程都通过了一个屏障点后,它会自动重置,可以再次被所有线程使用。但是,如果屏障被破坏,需要重新创建一个新的CyclicBarrier实例。
  • 4
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
CyclicBarrierJava 的一个同步工具类,它允许一组线程互相等待,直到所有线程都到达某个屏障点,然后再一起继续执行。CyclicBarrier 可以循环使用,因此称之为循环屏障。 下面是 CyclicBarrier 工具类的基本使用示例: ```java import java.util.concurrent.BrokenBarrierException; import java.util.concurrent.CyclicBarrier; public class CyclicBarrierDemo { public static void main(String[] args) { // 创建一个 CyclicBarrier 实例,指定等待线程数和屏障动作 CyclicBarrier cyclicBarrier = new CyclicBarrier(3, () -> { System.out.println("所有线程已到达屏障点,开始执行屏障动作"); }); // 创建 3 个线程,模拟多个线程同时到达屏障点 for (int i = 0; i < 3; i++) { new Thread(() -> { try { System.out.println(Thread.currentThread().getName() + " 已到达屏障点,等待其他线程"); cyclicBarrier.await(); // 等待其他线程到达屏障点 System.out.println(Thread.currentThread().getName() + " 继续执行"); } catch (InterruptedException | BrokenBarrierException e) { e.printStackTrace(); } }).start(); } } } ``` 上述代码,我们创建了一个 CyclicBarrier 实例,指定等待线程数和屏障动作。然后创建了 3 个线程,模拟多个线程同时到达屏障点,每个线程到达屏障点后等待其他线程,直到所有线程到达屏障点,才会继续执行。 在 CyclicBarrier 的构造函数,我们指定了等待线程数为 3,也就是说,只有当 3 个线程都到达屏障点时,才会执行屏障动作。屏障动作是一个 Runnable 对象,它会在所有线程到达屏障点后执行一次。 在每个线程执行的代码,我们调用了 CyclicBarrier 的 await() 方法,等待其他线程到达屏障点。当所有线程都到达屏障点后,await() 方法会返回,线程继续执行。 需要注意的是,CyclicBarrier 的 await() 方法可能会抛出 InterruptedException 和 BrokenBarrierException 异常,因此需要在 catch 块处理这两个异常。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值