热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

java并发工具类之CyclicBarrier

一.介绍CyclicBarrier的字面意思是可以循环使用的(Cyclic)的屏障(Barrier),它主要

一.介绍

CyclicBarrier的字面意思是可以循环使用的(Cyclic)的屏障(Barrier),它主要做的事情是,让一组线程到达一个屏障(也可以叫同步点)时被阻塞,直到最后一个线程到达屏障时,屏障才会开门,所有被屏障拦截的线程才会继续运行。

图01  :01与02 线程先到达了Barries,被阻塞,等待03线程的到来。

图02 :03 线程到达,Barries开门,02线程获得cpu执行权,先执行。

二. 使用

2.1 CyclicBarrier(int parties)使用

CyclicBarrier(int parties) 构造一个参数,参数代表屏障拦截线程数量。每个线程使用await方法来告诉CyclicBarries线程已经到达屏障,然后对该线程进行阻塞。

public class CyclicBarriesTest {public static CyclicBarrier barrier=new CyclicBarrier(2);public static void main(String[] args) {new Thread(()->{try {int await = barrier.await();} catch (InterruptedException e) {e.printStackTrace();} catch (BrokenBarrierException e) {e.printStackTrace();}System.out.println(Thread.currentThread().getName());},"t1").start();try {barrier.await();} catch (InterruptedException e) {e.printStackTrace();} catch (BrokenBarrierException e) {e.printStackTrace();}System.out.println(Thread.currentThread().getName());}
}

简单使用一下,创建CyclicBarries对象,然后线程数量设置成2. 在main方法中创建一个线程,在main函数与新建线程中分别调用await方法。如果你把main方法调用await方法之前让线程睡2s,你可以发现,线程t1与主线程的打印都延长

2.2  CyclicBarrier(int parties, Runnable barrierAction)

双参数构造,第一个参数是阻塞线程数量,第二个参数是所有线程到达屏障的时候,准备释放所有线程之前执行的一个动作(优先执行)。


public class CyclicBarriesTest02 {public static CyclicBarrier barrier =new CyclicBarrier(2, new Runnable() {@Overridepublic void run() {System.out.println("actionRun");}});public static void main(String[] args) {new Thread(()->{try {barrier.await();} catch (InterruptedException e) {e.printStackTrace();} catch (BrokenBarrierException e) {e.printStackTrace();}System.out.println(Thread.currentThread().getName());},"t1").start();try {barrier.await();} catch (InterruptedException e) {e.printStackTrace();} catch (BrokenBarrierException e) {e.printStackTrace();}System.out.println(Thread.currentThread().getName());}
}

执行结果:

actionRun
t1
main

三.使用场景

CyclicBarrier可以应用在多线程计算数据,最后合并计算结果的场景。比如我有个Excle保存了我一年的消费流水,然后一个月份一个sheet页,我想计算下我前三个月共消费多少。我就可以使用三个线程算出每个月的消费,最后在汇总。


public class CyclicBarrierTest3 {public static CyclicBarrier barrier &#61;new CyclicBarrier(3, new Runnable() {&#64;Overridepublic void run() {// 汇总Long count&#61; 0L;for (Map.Entry e :map.entrySet()) {count&#61;count&#43;e.getValue();}System.out.println("汇总&#xff1a;"&#43;count);}});public static ConcurrentMap map &#61;new ConcurrentHashMap<>();public static Executor executor &#61; Executors.newFixedThreadPool(3);// 使用3个线程算三个月的sheetpublic static void main(String[] args) {for (int i &#61; 0;i<3;&#43;&#43;i){executor.execute(()->{map.put(Thread.currentThread().getName(),System.currentTimeMillis());try {barrier.await();} catch (InterruptedException e) {e.printStackTrace();} catch (BrokenBarrierException e) {e.printStackTrace();}});}}}

使用三个线程计算出3个sheet页里的数据&#xff0c;最后使用CyclicBarrier 构造参数action 来汇总数据。输出结果如下&#xff1a;

汇总&#xff1a;4695037414899

注&#xff1a;本文章参考《java并发编程的艺术》&#xff0c;感谢方老师作品。


推荐阅读
  • 2023年京东Android面试真题解析与经验分享
    本文由一位拥有6年Android开发经验的工程师撰写,详细解析了京东面试中常见的技术问题。涵盖引用传递、Handler机制、ListView优化、多线程控制及ANR处理等核心知识点。 ... [详细]
  • 数据库内核开发入门 | 搭建研发环境的初步指南
    本课程将带你从零开始,逐步掌握数据库内核开发的基础知识和实践技能,重点介绍如何搭建OceanBase的开发环境。 ... [详细]
  • 本文详细介绍了 Dockerfile 的编写方法及其在网络配置中的应用,涵盖基础指令、镜像构建与发布流程,并深入探讨了 Docker 的默认网络、容器互联及自定义网络的实现。 ... [详细]
  • 本文介绍了如何使用 Spring Boot DevTools 实现应用程序在开发过程中自动重启。这一特性显著提高了开发效率,特别是在集成开发环境(IDE)中工作时,能够提供快速的反馈循环。默认情况下,DevTools 会监控类路径上的文件变化,并根据需要触发应用重启。 ... [详细]
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 本文介绍了Java并发库中的阻塞队列(BlockingQueue)及其典型应用场景。通过具体实例,展示了如何利用LinkedBlockingQueue实现线程间高效、安全的数据传递,并结合线程池和原子类优化性能。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • Android 渐变圆环加载控件实现
    本文介绍了如何在 Android 中创建一个自定义的渐变圆环加载控件,该控件已在多个知名应用中使用。我们将详细探讨其工作原理和实现方法。 ... [详细]
  • 本文探讨了如何在给定整数N的情况下,找到两个不同的整数a和b,使得它们的和最大,并且满足特定的数学条件。 ... [详细]
  • 本文详细介绍了 Apache Jena 库中的 Txn.executeWrite 方法,通过多个实际代码示例展示了其在不同场景下的应用,帮助开发者更好地理解和使用该方法。 ... [详细]
  • 本文详细介绍了Java中的访问器(getter)和修改器(setter),探讨了它们在保护数据完整性、增强代码可维护性方面的重要作用。通过具体示例,展示了如何正确使用这些方法来控制类属性的访问和更新。 ... [详细]
  • 本文介绍如何使用Objective-C结合dispatch库进行并发编程,以提高素数计数任务的效率。通过对比纯C代码与引入并发机制后的代码,展示dispatch库的强大功能。 ... [详细]
  • 深入理解 SQL 视图、存储过程与事务
    本文详细介绍了SQL中的视图、存储过程和事务的概念及应用。视图为用户提供了一种灵活的数据查询方式,存储过程则封装了复杂的SQL逻辑,而事务确保了数据库操作的完整性和一致性。 ... [详细]
  • 在前两篇文章中,我们探讨了 ControllerDescriptor 和 ActionDescriptor 这两个描述对象,分别对应控制器和操作方法。本文将基于 MVC3 源码进一步分析 ParameterDescriptor,即用于描述 Action 方法参数的对象,并详细介绍其工作原理。 ... [详细]
  • 扫描线三巨头 hdu1928hdu 1255  hdu 1542 [POJ 1151]
    学习链接:http:blog.csdn.netlwt36articledetails48908031学习扫描线主要学习的是一种扫描的思想,后期可以求解很 ... [详细]
author-avatar
梦幻死灵_791
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有