7.7 CyclicBarrier

CyclicBarrier 循环栅栏,用来进行线程协作,等待线程满足某个计数。构造时设置【计数个数】。每个线程执行到某个需要“同步”的时刻调用 await() 方法进行等待,当等待的线程数满足【计数个数】时,继续执行。

@Slf4j(topic = "c.TestCyclicBarrier")
public class TestCyclicBarrier {public static void main(String[] args) {ExecutorService executorService = Executors.newFixedThreadPool(2);CyclicBarrier barrier = new CyclicBarrier(2, () -> {log.debug("task1 task2 finish...");});for (int i = 0; i < 3; i++) {executorService.submit(() -> {log.debug("task1 begin...");sleep(1000);try {barrier.await();log.debug("task1 end...");} catch (InterruptedException | BrokenBarrierException e) {throw new RuntimeException(e);}});executorService.submit(() -> {log.debug("task2 begin...");sleep(2000);try {barrier.await();log.debug("task2 end...");} catch (InterruptedException | BrokenBarrierException e) {throw new RuntimeException(e);}});}}public static void sleep(long time) {try {Thread.sleep(time);} catch (InterruptedException e) {throw new RuntimeException(e);}}
}

输出

19:31:31.163 [pool-1-thread-1] - task1 begin...
19:31:31.163 [pool-1-thread-2] - task2 begin...
19:31:33.170 [pool-1-thread-2] - task1 task2 finish...
19:31:33.170 [pool-1-thread-2] - task2 end...
19:31:33.170 [pool-1-thread-1] - task1 end...
19:31:33.170 [pool-1-thread-2] - task1 begin...
19:31:33.170 [pool-1-thread-1] - task2 begin...
19:31:35.175 [pool-1-thread-1] - task1 task2 finish...
19:31:35.176 [pool-1-thread-1] - task2 end...
19:31:35.176 [pool-1-thread-1] - task1 begin...
19:31:35.176 [pool-1-thread-2] - task1 end...
19:31:35.176 [pool-1-thread-2] - task2 begin...
19:31:37.176 [pool-1-thread-2] - task1 task2 finish...
19:31:37.177 [pool-1-thread-2] - task2 end...
19:31:37.177 [pool-1-thread-1] - task1 end...
  • task1 需等待 task2 执行结束,task1 才能结束
  • CyclicBarrier 与 CountDownLatch 的主要区别在于 CyclicBarrier 是可以重用的 CyclicBarrier 可以被比喻为『人满发车』
  • 线程池中的线程数与 CyclicBarrier 中的 parties(第一个参数) 的数量最好保持一致。
@Slf4j(topic = "c.TestCyclicBarrier")
public class TestCyclicBarrier {public static void main(String[] args) {// 线程数与 parties 数量未保持一致ExecutorService executorService = Executors.newFixedThreadPool(3);CyclicBarrier barrier = new CyclicBarrier(2, () -> {log.debug("task1 task2 finish...");});// task1 执行时间为 1s,task2 执行时间为 2s// 可能会出现线程1,线程2与线程3同时执行,其中线程1和线程3执行 task1,// 线程2执行 task2。那么线程 1 和线程 3 在执行完 task1 后 CyclicBarrier // 的计数就减为了0,然后就去调用 CyclicBarrier 中的后续任务(参数2)for (int i = 0; i < 3; i++) {executorService.submit(() -> {log.debug("task1 begin...");sleep(1000);try {barrier.await();log.debug("task1 end...");} catch (InterruptedException | BrokenBarrierException e) {throw new RuntimeException(e);}});executorService.submit(() -> {log.debug("task2 begin...");sleep(2000);try {barrier.await();log.debug("task2 end...");} catch (InterruptedException | BrokenBarrierException e) {throw new RuntimeException(e);}});}}public static void sleep(long time) {try {Thread.sleep(time);} catch (InterruptedException e) {throw new RuntimeException(e);}}
}

输出

19:40:04.736 [pool-1-thread-2] - task2 begin...
19:40:04.736 [pool-1-thread-3] - task1 begin...
19:40:04.736 [pool-1-thread-1] - task1 begin...
19:40:05.742 [pool-1-thread-1] - task1 task2 finish...
19:40:05.742 [pool-1-thread-1] - task1 end...
19:40:05.743 [pool-1-thread-3] - task1 end...
19:40:05.743 [pool-1-thread-1] - task2 begin...
19:40:05.743 [pool-1-thread-3] - task1 begin...
19:40:06.748 [pool-1-thread-3] - task1 task2 finish...
19:40:06.748 [pool-1-thread-3] - task1 end...
19:40:06.748 [pool-1-thread-3] - task2 begin...
19:40:06.748 [pool-1-thread-2] - task2 end...
19:40:08.752 [pool-1-thread-3] - task1 task2 finish...
19:40:08.753 [pool-1-thread-3] - task2 end...
19:40:08.753 [pool-1-thread-1] - task2 end...

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/45794.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MySQL学习记录 —— 십구 命令执行

文章目录 1、mysql客户端命令2、从.sql文件执行SQL语句 1、mysql客户端命令 使用mysql时&#xff0c;命令行要以分号或者\g&#xff0c;\G来结束。 mysql客户端的很多命令可以在之前的mysql博客中查看。另外的用help或\h命令来查看。本篇写一些之前没有提到过的。 命令结束符…

先天睡功-守一老师

描述 守一老师&#xff0c;一个富有才华的老师&#xff01; 对于大家的学习有不可多得的帮助。 内容 目前主要的内容以睡觉为主&#xff0c;对于学习睡睡觉有比较大的帮助&#xff01; 但是网络上面错综复杂&#xff0c;很多老旧的版本影响学习&#xff01; 而这里我整理了…

安全防御实验2

一、实验拓扑 二、实验要求 办公区设备可以通过电信链路和移动链路上网(多对多的NAT&#xff0c;并且需要保留一个公网IP不能用来转换)分公司设备可以通过总公司的移动链路和电信链路访问到Dmz区的http服务器多出口环境基于带宽比例进行选路&#xff0c;但是&#xff0c;办公区…

数据建设实践之大数平台(六)安装spark

安装spark 上传安装包到/opt/software目录并解压 [bigdatanode101 software]$ tar -xvf spark-3.3.1-bin-hadoop3.tgz -C /opt/services/ [bigdatanode101 software]$ tar -xvf spark-3.3.1-bin-without-hadoop.tgz -C /opt/services/ 重命名文件 [bigdatanode101 servic…

OZON夏季热卖产品有哪些,OZON夏季热卖新品

OZON平台在夏季的热卖产品种类繁多&#xff0c;涵盖了多个领域&#xff0c;主要包括但不限于以下几个方面&#xff0c;接下来看看OZON夏季热卖产品有哪些&#xff0c;OZON夏季热卖新品&#xff01;Top1 运动套装 Костюм спортивный Victorias Secret 商品id…

c++课后作业

把字符串转换为整数 int main() {char pn[21];cout << "请输入一个由数字组成的字符串&#xff1a; ";cin >> pn;int last 0;int res[10];int j strlen(pn);int idx 2;cout << "请选择&#xff08;2-二进制&#xff0c;10-十进制&#xf…

【C++】C++入门实战教程(打造属于自己的C++知识库)

目录 目录 写在前面 1.C学习路线 2.本教程框架介绍 一.C基础部分 1.程序编码规范 2.程序运行与编译 3.关键字 4.常用数据类型 5.运算符相关 二.C进阶部分 1.面向对象编程 2.函数编程 3.模板编程 4.多线程与并发 5.STL介绍及使用 6.内存模型与优化 三.C实战部…

C++常用算法的简单总结

1、遍历算法 for_each(iterator beg, iterator end, func) :遍历容器 transform(iterator beg1, iterator end1, iterator beg2, _func): func可以直接搬运数据&#xff0c;也可以数据加减乘除之后搬运 2、查找算法 find(iterator beg, iterator end, 需要查找的数据) 查找元素…

美国视觉AI解决方案公司Hayden AI完成9000万美元C轮融资

来源&#xff1a;猛兽财经 作者&#xff1a;猛兽财经 猛兽财经获悉&#xff0c;总部位于美国加利福尼亚州旧金山弗朗西斯科专门为智慧城市提供视觉AI解决方案的Hayden AI&#xff0c;近期宣布已完成9000万美元C轮融资。 本轮融资由The Rise Fund领投&#xff0c;Drawdown Fun…

为什么go语言里从前端接收到的参数是数字28546.123456,但是不能使用float32只能使用float64呢?

在 Go 语言中&#xff0c;当从前端&#xff08;例如通过 HTTP 请求&#xff09;接收数据时&#xff0c;这些数据通常以字符串的形式到达后端。然后&#xff0c;后端需要将这些字符串解析或转换为适当的类型&#xff0c;比如 float32 或 float64。 然而&#xff0c;如果发现你只…

股指期货存在的风险有哪些?

股指期货因其标的物的特殊性&#xff0c;其面临的风险类型十分复杂&#xff0c;主要面临的一般风险和特有风险如下&#xff1a; 一般风险 从风险是否可控的角度&#xff0c;可以划分为不可控风险和可控风险&#xff1b;从交易环节可分为代理风险、流动性风险、强制平仓风险&…

BUCK外围器件选型,输入电容,输出电容,电感,续流二极管

概述&#xff1a; 一般情况下&#xff0c;电源接口处会有大小不同的电容进行并联&#xff0c;大容量电容是为了防止自身产生干扰影响其他器件&#xff0c;所以叫去耦电容&#xff1b;小容量电容是为了其他高频干扰影响自身&#xff0c;所以叫旁路电容。当然这只是通常情况下。 …

将获取pose 服务拆分为两个服务

简单拆分 要将该代码拆分为两个服务&#xff0c;我们需要创建两个FastAPI应用。第一个服务&#xff08;我们可以称之为ImageCaptureService&#xff09;将负责捕获视频流中的图像&#xff0c;并将图像数据发送到第二个服务&#xff08;我们可以称之为PoseEstimationService&am…

深入剖析多个表left join on的执行步骤原理:实战案例解析与原理探讨

文章目录 文章导图前言初始化数据-建表两个表left jion多表-left jion on c.bidb.bid分析|执行步骤和结果理解 变形-修改c表数据变形1变形2 总结 多表-left jion on c.aida.aid分析执行步骤和结果理解 变形-修改c表数据变形1变形2 解答开头总结 Left join on系列文章测试一下你…

ubuntu添加软件快捷方式

# 该目录下存放着所有的快捷方式文件 cd /usr/share/applications/ # 创建一个快捷方式文件 sudo touch your_app_name.desktop # 编辑快捷方式文件 sudo gedit your_app_name.desktop把&#x1f447;的内容复制进your_app_name.desktop文件中&#xff0c;这是最简化版本 [Des…

数据结构第24节 二分查找

二分查找&#xff08;Binary Search&#xff09;&#xff0c;也被称为折半查找&#xff0c;是一种在有序数组中查找特定元素的高效算法。它的基本思想是将查找区间分为两部分&#xff0c;通过比较中间元素与目标值来决定下一步是在哪一半继续查找。 二分查找的步骤&#xff1a…

​Chrome 插件: GoFullPage 一键搞定全网页截图

在互联网时代&#xff0c;网页截图已成为我们日常工作和生活中不可或缺的部分。无论是保存重要信息、制作教程&#xff0c;还是分享有趣的内容&#xff0c;截图功能都显得尤为重要。然而&#xff0c;常规的截图工具往往只能截取当前屏幕的内容&#xff0c;对于长网页则显得力不…

做个简单的知识付费网站需要什么方式

网站是线上承载信息宣传的主要工具之一&#xff0c;也是企业公司发展的重要工具之一&#xff0c;除了固定信息呈现外&#xff0c;还有不少商家具备各种方式的干货输出能力&#xff0c;或者想以内容售卖获得一定营收。 如教培机构、自媒体、网校、知识生产者、领域达人等都具备…

构建艺术:在Gradle中定制化输出的精粹

构建艺术&#xff1a;在Gradle中定制化输出的精粹 引言 Gradle是一个高度可配置的构建自动化工具&#xff0c;它广泛应用于现代软件开发中。在构建过程中&#xff0c;合理地配置构建输出对于项目的构建效率、部署和维护至关重要。本文将深入探讨如何在Gradle中配置构建输出&a…

【unity笔记】九、Unity添加串口通信

unity仿真使用虚拟串口调试。下面为简单流程。 常用串口调试软件在这里下载。 1.虚拟串口 添加虚拟串口&#xff0c;这里使用com1 com2 2. 串口调试 在这里为虚拟串口发送消息。 3. unity配置 3.1 设置 在文件->生成设置->玩家设置->玩家->其他设置 中找到…