Java多线程同步工具类CyclicBarrier的使用
作者:juer 发布时间:2021-08-26 05:29:53
CyclicBarrier是什么
CyclicBarrier是Java并发包中提供的一种同步工具类,它可以让多个线程在某个屏障处等待,直到所有线程都到达该屏障处才继续执行。CyclicBarrier的实现原理是基于ReentrantLock和Condition实现的,通过多次调用await()方法来实现线程的等待和唤醒。
CyclicBarrier的基本使用方法
CyclicBarrier的基本使用方法非常简单,只需要创建一个CyclicBarrier对象,并将计数器的值设置为等待的线程数。每个线程执行完毕后,调用CyclicBarrier的await()方法等待其他线程执行完毕,当所有线程都到达屏障处时,屏障将被打开,所有线程将继续执行。
CyclicBarrier的源码实现
CyclicBarrier的实现原理是基于ReentrantLock和Condition实现的,通过多次调用await()方法来实现线程的等待和唤醒。CyclicBarrier的源码实现主要包括两部分:屏障的初始化和屏障的等待和唤醒。
(1)CyclicBarrier的初始化
在创建CyclicBarrier对象时,需要指定等待的线程数和屏障的执行操作。CyclicBarrier对象的构造方法如下:
public CyclicBarrier(int parties, Runnable barrierAction)
其中,parties表示等待的线程数,barrierAction表示屏障执行的操作。
在构造方法中,会根据等待的线程数创建一个parties大小的ReentrantLock数组和一个Condition对象。ReentrantLock数组用来保证多个线程能够同时到达屏障处并等待,Condition对象用来进行线程的等待和唤醒。
(2)CyclicBarrier的等待和唤醒
当线程执行到await()方法时,会首先尝试获取ReentrantLock对象的锁,如果获取失败,线程会被加入到等待队列中等待锁的释放。当获取到锁后,线程会判断当前的计数器是否已经达到等待的线程数,如果是,则执行屏障的操作并将计数器重置为parties,唤醒等待队列中的所有线程。如果计数器未达到等待的线程数,则线程会被加入到等待队列中等待其他线程的到来。
CyclicBarrier的await()方法源码如下:
public int await() throws InterruptedException, BrokenBarrierException {
try {
// 获取锁
lock.lock();
// 计数器减1
int index = --count;
if (index == 0) {
// 如果计数器为0,执行屏障操作并唤醒等待队列中的所有线程
final Runnable command = barrierCommand;
if (command != null) {
command.run();
}
next
trip.signalAll();
} else {
try {
// 等待其他线程到达屏障处
int phase = generation;
trip.await();
// 如果是最后一个到达屏障的线程,执行屏障操作并唤醒等待队列中的所有线程
if (phase == generation) {
command = barrierCommand;
if (command != null) {
command.run();
}
}
// 计数器重置
nextGeneration();
} catch (InterruptedException ie) {
// 如果线程在等待时被中断,抛出InterruptedException异常
cancel();
throw ie;
} catch (BrokenBarrierException bbe) {
// 如果屏障被破坏,抛出BrokenBarrierException异常
broken = true;
trip = new Condition[parties];
throw bbe;
}
}
return index;
} finally {
// 释放锁
lock.unlock();
}
}
在CyclicBarrier的await()方法中,首先获取ReentrantLock对象的锁,并将计数器减1。如果计数器为0,则执行屏障的操作并唤醒等待队列中的所有线程,如果计数器不为0,则等待其他线程到达屏障处。
在等待过程中,如果线程被中断,将抛出InterruptedException异常。如果屏障被破坏,将抛出BrokenBarrierException异常。如果是最后一个到达屏障的线程,将执行屏障的操作并唤醒等待队列中的所有线程,并将计数器重置为parties。
CyclicBarrier的使用场景
CyclicBarrier适用于多个线程需要等待彼此到达某个屏障点后再继续执行的场景。例如,多个线程需要同时执行某个任务,但某个任务需要等待其他任务完成后才能继续执行,这时就可以使用CyclicBarrier来实现线程的同步和协作。
另外,CyclicBarrier也可以用来实现流水线式的处理,例如生产者消费者模式中,多个生产者可以同时向队列中添加数据,当队列满时,所有生产者需要等待消费者处理完数据后再继续添加数据。
来源:https://juejin.cn/post/7226448512627589157


猜你喜欢
- 排列组合的概念排列:从n个不同元素中取出m(m≤n)个元素,按照一定的顺序排成一列,叫做从n个元素中取出m个元素的一个排列(Arrangem
- 如果对一个值可以包含多个,那么可以使用枚举,加上Flags。新建一个Flags枚举类型:[Flags] publi
- 1.背景Java语言相比于C和C++,一个最大的特点就是不需要程序员自己手动去申请和释放内存,这一切交由JVM来完成。在Java中,运行时的
- openid可以标识一个用户,session_key会变,所以来获取一下openid。openid不能在微信小程序中直接获取,需要后台发送请
- 1、获取表中最后一条数据public static String demo() throws SQLException { String s
- 简介Spring Security,这是一种基于 Spring AOP 和 Servlet 过滤器的安全框架。它提供全面的安全性解决方案,同
- 目录文件下载文件上传 * * 的配置多个 * 的执行顺序异常处理器基于配置的异常处理基于注解的异常处理总结文件下载使用ResponseEn
- 先说明一下,项目代码已上传至github,不想看长篇大论的也可以先去下代码,对照代码,哪里不懂点哪里。代码在这https://github.
- 向量向量是序列容器,表示可以更改大小的数组。就像数组一样,向量对其元素使用连续的存储位置,这意味着也可以使用指向其元素的常规指针上的偏移量来
- 步骤,如图所示:1.添加异步任务业务类package top.ytheng.demo.task;import java.util.concu
- 前言基于安卓平台的连续滚动图像组件ContinuousScrollableImageView(https://github.com/Cutt
- 本文实例为大家分享了Android实现接近传感器的具体代码,供大家参考,具体内容如下1.接近传感器检测物体与听筒(手机)的距离,单位是厘米。
- 同步器简介 学习以来对线程的操作有很大的改观,从c/c++的mute
- 本文为大家分享了类似微信朋友圈,点击+号图片,可以加图片功能,供大家参考,具体内容如下xml:<?xml version="
- 本文实例为大家分享了Unity实现全屏截图、Unity实现QQ截图,供大家参考,具体内容如下全屏截图:要实现的是点击鼠标左键,就实现截图,并
- 本文实例为大家分享了C语言实现餐饮管理系统的具体代码,供大家参考,具体内容如下#include <stdio.h>#includ
- 上一章说明了DataBinding生存的类之间关系,现在这里来看看布局是如何加载的以及view是如何映射的。一、布局加载这里把之前的代码重新
- 概述什么是 Spring WebFlux, 它是一种异步的, 非阻塞的, 支持背压(Back pressure)机制的Web 开发框架. 要
- 前言在分布式场景下为了保证数据最终一致性。在单进程的系统中,存在多个线程可以同时改变某个变量(可变共享变量)时,就需要对变量或代码块做同步(
- java 基础之JavaBean属性命名规范问题JavaBean属性名要求:前两个字母要么都大写,要么都小写下面我们来找找如果不遵循这个规范