Java高并发之CyclicBarrier的用法详解
作者:xindoo 发布时间:2023-11-17 16:27:57
Java 中的 CyclicBarrier 是一种同步工具,它可以让多个线程在一个屏障处等待,直到所有线程都到达该屏障处后,才能继续执行。CyclicBarrier 可以用于协调多个线程的执行,以便它们可以在某个点上同步执行。
CyclicBarrier 是 Java 中的一种同步工具,它可以让多个线程在一个屏障点处等待,直到所有线程都到达该点后,才能继续执行。CyclicBarrier 可以用于协调多个线程的执行,以便它们可以在某个点上同步执行。
使用方式
CyclicBarrier 的基本用法如下:
import java.util.concurrent.BrokenBarrierException;
import java.util.concurrent.CyclicBarrier;
public class CyclicBarrierExample {
public static void main(String[] args) {
int n = 3;
CyclicBarrier barrier = new CyclicBarrier(n, new Runnable() {
public void run() {
System.out.println("All threads have reached the barrier");
}
});
Thread t1 = new Thread(new MyRunnable(barrier), "Thread 1");
Thread t2 = new Thread(new MyRunnable(barrier), "Thread 2");
Thread t3 = new Thread(new MyRunnable(barrier), "Thread 3");
t1.start();
t2.start();
t3.start();
}
static class MyRunnable implements Runnable {
private final CyclicBarrier barrier;
public MyRunnable(CyclicBarrier barrier) {
this.barrier = barrier;
}
public void run() {
try {
System.out.println(Thread.currentThread().getName() + " is waiting at the barrier...");
barrier.await();
System.out.println(Thread.currentThread().getName() + " has crossed the barrier");
} catch (InterruptedException e) {
e.printStackTrace();
} catch (BrokenBarrierException e) {
e.printStackTrace();
}
}
}
}
在这个例子中,我们创建了一个 CyclicBarrier 对象,它需要等待 3 个线程到达屏障点。当所有线程都到达屏障点后,将会触发一个回调函数,打印一条消息。
我们创建了 3 个线程,并将它们传递给一个自定义的 Runnable 对象。在每个线程的 run 方法中,我们首先打印一条消息,表示线程正在等待屏障点。然后调用 barrier.await() 方法,将线程加入到等待队列中,直到所有线程都到达屏障点后,才会继续执行。在最后,我们打印一条消息,表示线程已经跨过了屏障点。
上面代码的运行结果如下:
Thread 1 is waiting at the barrier...
Thread 3 is waiting at the barrier...
Thread 2 is waiting at the barrier...
All threads have reached the barrier
Thread 2 has crossed the barrier
Thread 1 has crossed the barrier
Thread 3 has crossed the barrier
从上面代码中也可以看出,CyclicBarrier 还支持一个可选的回调函数,在所有的线程都到达屏障点后,会调起指定的回调函数,上述例子中当所有线程到达屏障点的时候,会执行回调函数,表明已经到达屏障点。
CyclicBarrier 还支持一个更高级的用法,即可以在等待线程到达屏障点时,执行一些额外的操作。可以通过 await 方法的返回值来实现这一点,如下所示:
int index = barrier.await();
if (index == 0) {
// 执行额外的操作
}
在这个例子中,await 方法的返回值表示线程在等待队列中的位置,如果返回值为 0,则表示当前线程是最后一个到达屏障点的线程,可以执行一些额外的操作,比如说做一些数据清理之类的收尾工作。
注意事项
在使用 Java 中的 CyclicBarrier 时,需要注意以下几点:
1.CyclicBarrier 的计数器是可重用的,也就是说,当所有线程都到达屏障点后,计数器会被重置为初始值,可以再次使用。如果在等待过程中出现异常,计数器将会被重置,并且所有等待的线程都将会抛出 BrokenBarrierException 异常。
2.如果使用 CyclicBarrier 时,等待的线程数超过了计数器的初始值,将会导致所有线程永远等待下去。因此,在使用 CyclicBarrier 时,需要确保等待的线程数不会超过计数器的初始值。
3.CyclicBarrier 的回调函数是在最后一个线程到达屏障点时执行的,因此,在回调函数中执行的操作应该是线程安全的,否则可能会导致不可预期的结果。
4.CyclicBarrier 可以用于协调多个线程的执行,以便它们可以在某个点上同步执行。**但是,如果线程之间的执行顺序对于程序的正确性很重要,那么 CyclicBarrier 可能不是最好的选择。**在这种情况下,可能需要使用其他同步工具,如 CountDownLatch 或 Semaphore。
5.CyclicBarrier 的性能可能会受到等待线程的数量和计数器的初始值的影响。**如果等待线程的数量很大,或者计数器的初始值很大,那么可能会导致性能下降。**因此,在使用 CyclicBarrier 时,需要根据实际情况进行调整。
总之,在使用 Java 中的 CyclicBarrier 时,需要仔细考虑各种情况,以确保程序的正确性和性能。
来源:https://juejin.cn/post/7209617649885184058


猜你喜欢
- forword跳转页面的三种方式:1.使用serlvet/** * 使用forward跳转,传递基本类型参数到页面
- @PostConstruct不被调用的原因如果在配置文件中配置使用,延迟加载的话如图被@Service等注解的类,需要在注入使用的时候,才会
- 本文实例讲述了WPF设置窗体可以使用鼠标拖动大小的方法。分享给大家供大家参考。具体实现方法如下:private void Window_Lo
- 问题:对于多线程编程,很多时候往往需要向线程中传递多个参数,而C#中的线程只接收1个object类型的参数(如下):Thread t = n
- 本文实例为大家分享了Unity Shader实现动态过场切换图片的具体代码,供大家参考,具体内容如下一、简单介绍Shader Languag
- MD5,全称为 Message Digest Algorithm 5(消息摘要算法第五版).详情请参考 * :MD5MD5加密后是一个字
- 在使用一些产品列如微信、QQ之类的,如果有新消息来时,手机屏幕即使在锁屏状态下也会亮起并提示声音,这时用户就知道有新消息来临了。但是,一般情
- 1 前言许多语言,例如 Perl ,Python 和 Ruby ,都有集合的本地支持。有些语言(例如Python)甚至将基本集合组件(列表,
- 析构函数用于析构类的实例。备注不能在结构中定义析构函数。只能对类使用析构函数。一个类只能有一个析构函数。无法继承或重载析构函数。无法调用析构
- 本文实例讲述了Android AutoCompleteTextView连接数据库自动提示的方法。分享给大家供大家参考,具体如下:这个简单例子
- 分页使用可以说非常普遍了,有时候会需要非常灵活的方式去开启或关闭分页,尝试使用一 * 解的方式来进行分页。依赖安装需要使用的依赖:Mybati
- 由于一个线程的程序,如果调用一个功能是阻塞的,那么就会影响到界面的更新,导致使用人员操作不便。所以往往会引入双线程的工作的方式,主线程负责更
- SpringBoot项目中新增脱敏功能项目背景目前正在开发一个SpringBoot项目,此项目有Web端和微信小程序端。web端提供给工作人
- IoC——Inversion of Control,控制反转在Java开发中,IoC意味着将你设计好的类交给系统去控制,而不是在你的类内部控
- 哈喽大家好啊,我是Hydra。Spring作为项目中不可缺少的底层框架,提供的最基础的功能就是bean的管理了。bean的注入相信大家都比较
- Maven 组件界面介绍 如上图标注 1 所示,为常用的 Maven 工具栏,其中最常用的有:第一个按钮:Reimport All
- SpringMVC 中,文件的上传,是通过 MultipartResolver 实现的。 所以,如果要实现文件的上传,只要在 spring-
- Apache Dubbo是一款高性能、轻量级的开源 Java RPC 框架,它提供了三大核心能力:面向接口的远程方法调用,智能容错和负载均衡
- 在LINUX上部署带有JAR包的JAVA项目首先eclipse上要装上一个小插件,叫做Fat Jar点击Fat Jar红框里选上主类点击Ne
- 可以用如下方法: 修改AudioYusuStreamOut.cpp,添加方法: void AudioYusuStreamOut::swS2M