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

标签:Java,高并发,CyclicBarrier
0
投稿

猜你喜欢

  • Android编程常用技巧实例总结

    2022-11-30 20:27:44
  • win10下配置java环境变量的方法

    2022-08-23 17:10:11
  • JavaWeb建立简单三层项目步骤图解

    2023-03-08 16:51:02
  • 详解Java编程中protected修饰符与static修饰符的作用

    2022-05-11 03:53:17
  • Android SwipeRefreshLayout超详细讲解

    2023-08-22 13:43:28
  • Java数据结构之图的路径查找算法详解

    2023-03-27 03:47:11
  • spring入门教程之bean的继承与自动装配详解

    2023-11-10 14:46:23
  • Android PopupWindow被输入法弹上去之后无法恢复原位的解决办法

    2023-07-24 12:23:31
  • jenkins构建go及java项目的方法

    2022-09-26 12:39:01
  • java中JSONObject转换为HashMap(方法+main方法调用实例)

    2023-08-10 04:04:08
  • C#中DataGridView的样式

    2022-04-23 11:08:42
  • Android开发实现TextView超链接5种方式源码实例

    2022-12-10 16:50:32
  • python调用java的Webservice示例

    2021-07-31 13:36:09
  • 浅谈Java线程Thread之interrupt中断解析

    2023-07-19 09:25:11
  • Java_异常类(错误和异常,两者的区别介绍)

    2023-09-19 08:53:27
  • C#多线程之线程池ThreadPool用法

    2021-07-21 06:28:40
  • Jackson多态序列化图文详解

    2022-01-26 19:46:08
  • java中@SuppressWarnings注解用法详解

    2023-09-20 23:11:40
  • c# 实现语音合成

    2021-06-16 00:17:47
  • 使用SpringBoot配置多数据源的经验分享

    2022-05-25 04:02:57
  • asp之家 软件编程 m.aspxhome.com