JUC包:CyclicBarrier源码+实例讲解
1 缘起
上篇文章讲到了CountDownLatch:https://blog.csdn.net/Xin_101/article/details/129116170
作为同系的佼佼者,不得不提CyclicBarrier,
设计理念相似,都是多线程等待,但是,应用的技术以及功能不同,
下面根据源码及实例讲解,
帮助读者轻松应对知识交流与考核。
2 CyclicBarrier
同步辅助工具,允许一组线程相互等待对方达到共同的屏障点。
CyclicBarrier在固定大小的线程组中非常有用,这些线程组可以偶尔彼此等待。
这个屏障被称为循环屏障,因为释放等待线程后可以重新使用线程和屏障点。
CyclicBarrier支持Runnable(可选)命令,这个Runnable介于到达屏障点前和突破屏障点后,在每个屏障点执行一次。
Runnable屏障点操作对任何一方继续之前更新共享状态非常有用。
内存一致性影响:先调用await方法的线程动作发生在其他部分屏障操作之前(先于其他线程通过await方法获取响应结果)。
2.1 测试样例
使用CyclicBarrier有两个关键,初始化和await,
其中,初始化有两种方式,第一,使用Runnable,当到达屏障时执行;第二,不使用Runnable;
await线程等待,当同步状态为0时,唤醒线程,释放锁,突破屏障,继续执行后面的逻辑。
同时,CyclicBarrier是可重用的,先给一个测试的样例。
package com.monkey.java_study.juc;import org.apache.commons.lang3.time.StopWatch;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;import java.util.Random;
import java.util.concurrent.CyclicBarrier;/*** 测试CyclicBarrier.** @author xindaqi* @since 2023-02-20 15:48*/
public class CyclicBarrierTest {private static final Logger logger = LoggerFactory.getLogger(CyclicBarrierTest.class);static class ThreadRunner implements Runnable {private final CyclicBarrier cyclicBarrier;public ThreadRunner(CyclicBarrier cyclicBarrier) {this.cyclicBarrier = cyclicBarrier;}@Overridepublic void run() {Random random = new Random();int randomBound = 1000;try {// 第一个屏障StopWatch stopWatch = new StopWatch();stopWatch.start();Thread.sleep(random.nextInt(randomBound));stopWatch.stop();logger.info(">>>>>>>>{} 到达第一个屏障, time cost:{}", Thread.currentThread().getName(), stopWatch.formatTime());cyclicBarrier.await();logger.info(">>>>>>>>{} 突破第一个屏障, time cost:{}", Thread.currentThread().getName(), stopWatch.formatTime());// 第二个屏障stopWatch.reset();stopWatch.start();Thread.sleep(random.nextInt(randomBound));stopWatch.stop();logger.info(">>>>>>>>{} 到达第二个屏障, time cost:{}", Thread.currentThread().getName(), stopWatch.formatTime());cyclicBarrier.await();logger.info(">>>>>>>>{} 突破第二个屏障, time cost:{}", Thread.currentThread().getName(), stopWatch.formatTime());} catch (Exception ex) {throw new RuntimeException(ex);}}}public static void main(String[] args) {StopWatch stopWatch = new StopWatch();stopWatch.start();CyclicBarrier cyclicBarrier = new CyclicBarrier(3, () -> {logger.info(">>>>>>>>开始突破屏障.");});for (int i = 0; i < 3; i++) {new Thread(new ThreadRunner(cyclicBarrier)).start();}}
}

3 源码分析
先看下CyclicBarrier的两个属性,lock和trip,
源码如下图所示,其中,lock为ReentrantLock,trip为Condition,
由Condition可知,通过await可实现线程等待,释放锁,
CyclicBarrier通过这个技术实现线程等待,线程唤醒,实现屏障的功能。

3.1 初始化
先看CyclicBarrier初始化,源码如下图所示,
由图可知,参数有两个:同步状态数量和Runnable,
其中,同步状态数量表示等待线程的数量,Runnable用于到达屏障时执行逻辑。
位置:java.util.concurrent.CyclicBarrier#CyclicBarrier(int, java.lang.Runnable)

下面的初始化不指定Runnable,源码如下图所示,
由源码可知,只指定了同步状态数量,而Runnable则为null,后续不会执行其他逻辑。
位置:java.util.concurrent.CyclicBarrier#CyclicBarrier(int)

3.2 await
完成CyclicBarrier的初始化,
接下来需要维护屏障,开启线程等待,
通过await方法实现,源码如下图所示。
await有两种,带参和不带参,
不带参的源码如下图所示。
位置:java.util.concurrent.CyclicBarrier#await()

带参方法源码如下图所示,由图可知,
参数为超时时间,是线程等待的最大时间,线程等待超时后抛出异常。
具体的实现还要看dowait,后面接着分析。
位置:java.util.concurrent.CyclicBarrier#await(long, java.util.concurrent.TimeUnit)

3.2.1 dowait
dowait源码如下图所示,源码比较长,分段分析,
源码中标识了解析。
位置:java.util.concurrent.CyclicBarrier#dowait
第一段,源码如下图所示,
由源码可知,CyclicBarrier使用了可重入锁Reentrant,
一般而言,多线程使用ReentrantLock会阻塞,线程进入队列排队,
但是,正如ReentrantLock设计,对线程的使用提供了更高的灵活性,
可以通过await让线程等待,并释放锁,通过signalAll()唤醒等待的线程,
于是,CyclicBarrier多线程不会进入阻塞,而是线程进入等待状态,然后释放锁,
这样其他线程可以不用等待其他已经获得锁的线程释放锁就可以获取锁,因为其他线程进入等待状态并且释放了锁。
这部分的逻辑通过配置generation作为突破标识,
并且使同步状态减1,同步状态为0时,进入对应的逻辑,后面讲。
3.2.1.1 获取锁

3.2.1.2 冲破屏障
接下来进入同步状态为0的逻辑,源码如下图所示,
由图可知,
同步状态为0时,说明所有线程均已执行,并且最后一进入的线程之前的所有线程已经进入等待状态,
最后一个进入的线程执行同步状态减1后,
通过nextGeneration重置同步状态,更新generation,满足跳出自旋的条件,
并且使用signalAll唤醒所有线程,后面源码分析nextGeneration,
在finally中通过breakBarrier突破屏障,后面讲。

nextGeneration方法如下图所示,
由图可知,
通过newGeneration更新generation,
以满足跳出自旋的条件g != generation。
同时,通过signalAll唤醒所有线程,重置同步状态。

3.2.1.3 线程等待
接下来就是CyclicBarrier的等待逻辑,源码如下图所示,
由图可知,
通过自旋,实现线程相互等待,
当然,需要在完成任务后跳出自旋,通过generation作为突破屏障的标识,
当generation发生改变时,直接返回,跳出自旋,进入finally,突破屏障,
最后一个线程进入,使同步状态减为0,由前文可知,更新generation,
满足g != generation,跳出自旋,
最后,finally中的逻辑释锁。

4 小结
(1)CyclicBarrier通过ReentrantLock和Condition实现线程等待,释放锁,保证多线程不阻塞;
(2)以generation作为突破屏障的标志;
(3)线程间相互等待:通过自旋实现线程间相互等待,同步状态减为0时,更新generation,满足跳出自旋的条件,最后一个线程进入后,唤醒其余等待的线程,为后面重用做准备,并跳出当前自旋,突破屏障,释放锁;
(4)突破屏障:最后一个线程进入后,同步状态减为0,跳出自旋,突破屏障;
(5)CyclicBarrier线程可重用:通过重置同步状态,唤醒所有线程。
相关文章:
JUC包:CyclicBarrier源码+实例讲解
1 缘起 上篇文章讲到了CountDownLatch:https://blog.csdn.net/Xin_101/article/details/129116170 作为同系的佼佼者,不得不提CyclicBarrier, 设计理念相似,都是多线程等待,但是,应用的技术以及功能不同&a…...
Trace、Metrics、Logging 选型
背景分布式追踪的起源自从微服务的兴起开始,整个系统架构开始变得极为庞大和复杂,但是服务之间的调用关系,调用消耗时间等等信息却依然是半黑盒的状态。为了能够将调用的链路进行串联,将系统的各种指标数据展示出来以使得系统的链…...
Java验证码
文章目录一、验证码概述二、Java原生验证码1、随机数字验证码2、随机数字和字母验证码3、运算验证码三、引入三方验证码一、验证码概述 验证码(CAPTCHA)是“Completely Automated Public Turing test to tell Computers and Humans Apart”(全…...
5天带你读完《Effective Java》(四)
《Effective Java》是Java开发领域无可争议的经典之作,连Java之父James Gosling都说: “如果说我需要一本Java编程的书,那就是它了”。它为Java程序员提供了90个富有价值的编程准则,适合对Java开发有一定经验想要继续深入的程序员…...
探索密码学的未来:SM1、SM2、SM3、SM4、同态加密、密态计算、隐私计算和安全多方计算
密码算法在现代通信与信息安全中发挥着至关重要的作用,SM1、SM2、SM3、SM4、同态加密、密态计算、隐私计算和安全多方计算等密码算法被广泛应用于各种信息安全领域。本篇博客将会为大家介绍这些密码算法,以及它们在信息安全中的作用和应用。 一、SM1、SM…...
【教程】去水印开源工具Lama Cleaner在Windows的安装和使用
一、Lama Cleaner是什么? Lama Cleaner是一款开源且免费的人工学习图片去水印程序(个人主要学习用途),没有图片分辨率限制(个人使用暂未发现),并且保存的图片质量很高(个人觉得跟原…...
驾考笔记_2023
科目一1> 扣分制度1.1> 超速1.2> 超载1.3> 车牌1.4> 速记口诀2> 满分学习2.1> 消分学习2.2> 满分重考;3> 罚款 / 判刑3.1> 考证3.2> 审验教育3.3> 酒驾3.4> 200¥3.5> 500¥3.6> 2000¥…...
【架构师】跟我一起学架构——调用链
博客昵称:架构师Cool 最喜欢的座右铭:一以贯之的努力,不得懈怠的人生。 作者简介:一名Coder,软件设计师/鸿蒙高级工程师认证,在备战高级架构师/系统分析师,欢迎关注小弟! 博主小留言…...
[神经网络]Swin Transformer网络
一、概述 Swin Transformer是一个用了移动窗口的层级式Vision Transformer。 在图像领域,Transformer需要解决如下两个问题: ①尺度问题:同一语义的物体在图像中有不一样的尺度。(大小不同) ②Resolution过大:若以像素点作为单位&…...
【分布式】什么是分布式,分布式和集群的区别又是什么?答案在正文。
文章目录1. 什么是分布式 ?2. 分布式与集群的区别 ?3.用一个请求串起来4.一个简化的架构图5.分布式环境的特点6.分布式环境下面临的问题7.总结1. 什么是分布式 ? 分布式系统一定是由多个节点组成的系统。 其中,节点指的是计算机服务器,而且这些节点一…...
MyBatis框架的入门案例
MyBatis框架的入门案例 资源地址:https://download.csdn.net/download/weixin_41957626/87531373 1.MyBatis的配置 环境:基于maven的结构 1.1目录结构 1.2依赖包 <dependencies><!--mybatis--><dependency><groupId>org.mybatis…...
红黑树-随记
文章目录1.为什么hashmap用红黑树不用二叉树和平衡二叉树1.1 二叉树(Binary Search Tree)1.2 红黑树(Red Black Tree)1.3 平衡二叉树(Balence Binary Tree)也称AVT2.为什么mysql用b数,不用B数或…...
Python异常处理更新,正常和不正常的都在这里
嗨害大家好鸭!我是小熊猫~ 异常处理篇嗨害大家好鸭!我是小熊猫~Python标准异常💨什么是异常?不正常异常处理💨使用except而不带任何异常类型使用except而带多种异常类型try-finally 语句异常的参数触发异常用户自定义异…...
[数据结构]:10-二叉排序树(无头结点)(C语言实现)
目录 前言 已完成内容 二叉排序树实现 01-开发环境 02-文件布局 03-代码 01-主函数 02-头文件 03-BinarySearchTreeCommon.cpp 04-BinarySearchTreeFunction.cpp 结语 前言 此专栏包含408考研数据结构全部内容,除其中使用到C引用外,全为C语言…...
openstack浅析
** OpenStack是一个由多个组件组成的开源云计算平台,每个组件都有不同的功能和用途。 ** 组件构成 以下是OpenStack中一些常见的组件及其功能: Nova:用于管理虚拟机的组件,提供了虚拟机的创建、销毁、管理等功能。 Neutron&am…...
华为OD机试Golang解题 - 特异性双端队列 | 含思路
华为Od必看系列 华为OD机试 全流程解析+经验分享,题型分享,防作弊指南)华为od机试,独家整理 已参加机试人员的实战技巧华为od 2023 | 什么是华为od,od 薪资待遇,od机试题清单华为OD机试真题大全,用 Python 解华为机试题 | 机试宝典文章目录 华为Od必看系列使用说明本期题目…...
代码随想录中:回溯算法的基础
回溯算法是一种暴力的搜索方式;回溯法一般与递归同时存在。 回溯法,一般可以解决如下几种问题: 组合问题:N个数里面按一定规则找出k个数的集合切割问题:一个字符串按一定规则有几种切割方式子集问题:一个…...
Android kotlin 系列讲解(进阶篇)Jetpack系列之LiveData
<<返回总目录 文章目录 一、LiveData是什么二、LiveData测试一、LiveData是什么 LiveData是Jetpack提供的一种响应式编程组件,它可以包括任何类型的数据,并在数据发生变化的时候通知给观察者。LiveData特别适合与ViewModel结合在一起使用,虽然它也可以单独在别的地方…...
如何判断有向无环图:构造有向无环图
拓扑序列:可以用来判断一个有向图是否有环! 拓扑排序可以判断有向图是否存在环。我们可以对任意有向图执行上述过程,在完成后检查A序列的长度。 若A序列的长度小于图中点的数量,则说明某些节点未被遍历,进而说明图中存…...
【2022.1.3】手脱压缩壳练习(含练习exe)
【2022.1.3】手脱压缩壳练习(含练习exe) 文章目录【2022.1.3】手脱压缩壳练习(含练习exe)0、简介1、单步跟踪法(#)方法介绍(0)练习exe下载(1)、查看源程序&am…...
理解 MCP 工作流:使用 Ollama 和 LangChain 构建本地 MCP 客户端
🌟 什么是 MCP? 模型控制协议 (MCP) 是一种创新的协议,旨在无缝连接 AI 模型与应用程序。 MCP 是一个开源协议,它标准化了我们的 LLM 应用程序连接所需工具和数据源并与之协作的方式。 可以把它想象成你的 AI 模型 和想要使用它…...
学校招生小程序源码介绍
基于ThinkPHPFastAdminUniApp开发的学校招生小程序源码,专为学校招生场景量身打造,功能实用且操作便捷。 从技术架构来看,ThinkPHP提供稳定可靠的后台服务,FastAdmin加速开发流程,UniApp则保障小程序在多端有良好的兼…...
Python爬虫(二):爬虫完整流程
爬虫完整流程详解(7大核心步骤实战技巧) 一、爬虫完整工作流程 以下是爬虫开发的完整流程,我将结合具体技术点和实战经验展开说明: 1. 目标分析与前期准备 网站技术分析: 使用浏览器开发者工具(F12&…...
【git】把本地更改提交远程新分支feature_g
创建并切换新分支 git checkout -b feature_g 添加并提交更改 git add . git commit -m “实现图片上传功能” 推送到远程 git push -u origin feature_g...
用docker来安装部署freeswitch记录
今天刚才测试一个callcenter的项目,所以尝试安装freeswitch 1、使用轩辕镜像 - 中国开发者首选的专业 Docker 镜像加速服务平台 编辑下面/etc/docker/daemon.json文件为 {"registry-mirrors": ["https://docker.xuanyuan.me"] }同时可以进入轩…...
C++使用 new 来创建动态数组
问题: 不能使用变量定义数组大小 原因: 这是因为数组在内存中是连续存储的,编译器需要在编译阶段就确定数组的大小,以便正确地分配内存空间。如果允许使用变量来定义数组的大小,那么编译器就无法在编译时确定数组的大…...
【Go语言基础【12】】指针:声明、取地址、解引用
文章目录 零、概述:指针 vs. 引用(类比其他语言)一、指针基础概念二、指针声明与初始化三、指针操作符1. &:取地址(拿到内存地址)2. *:解引用(拿到值) 四、空指针&am…...
力扣热题100 k个一组反转链表题解
题目: 代码: func reverseKGroup(head *ListNode, k int) *ListNode {cur : headfor i : 0; i < k; i {if cur nil {return head}cur cur.Next}newHead : reverse(head, cur)head.Next reverseKGroup(cur, k)return newHead }func reverse(start, end *ListNode) *ListN…...
从面试角度回答Android中ContentProvider启动原理
Android中ContentProvider原理的面试角度解析,分为已启动和未启动两种场景: 一、ContentProvider已启动的情况 1. 核心流程 触发条件:当其他组件(如Activity、Service)通过ContentR…...
Qt 事件处理中 return 的深入解析
Qt 事件处理中 return 的深入解析 在 Qt 事件处理中,return 语句的使用是另一个关键概念,它与 event->accept()/event->ignore() 密切相关但作用不同。让我们详细分析一下它们之间的关系和工作原理。 核心区别:不同层级的事件处理 方…...
