1.CountDownLatch

          countDownLatch的作用是让一组线程等待其他线程完成工作以后在执行,相当于加强版的join(不懂可以百度一下join的用法),一般在初始化的时候会在构造方法传入计数器,

后续,在其他线程中每次调用countDown方法计数器减一,一般在需要等待的线程中调用countDownLatch的await方法阻塞线程,在当计数器为0时,等待线程继续运行。

光看上面的定义描述不是很直观,我们再来结合代码看一下实际运用:

 1 public class UseCountDownLatch {
2
3 static CountDownLatch latch = new CountDownLatch(6);
4 //初始化线程(只有一步,有4个)
5 private static class InitThread implements Runnable{
6
7 @Override
8 public void run() {
9 System.out.println("Thread_"+Thread.currentThread().getId()
10 +" ready init work......");
11 latch.countDown();//初始化线程完成工作了,countDown方法只扣减一次;
12 for(int i =0;i<2;i++) {
13 System.out.println("Thread_"+Thread.currentThread().getId()
14 +" ........continue do its work");
15 }
16 }
17 }
18 //业务线程
19 private static class BusiThread implements Runnable{
20
21 @Override
22 public void run() {
23 try {
24 latch.await();
25 } catch (InterruptedException e) {
26 e.printStackTrace();
27 }
28 for(int i =0;i<3;i++) {
29 System.out.println("BusiThread_"+Thread.currentThread().getId()
30 +" do business-----");
31 }
32 }
33 }
34
35 public static void main(String[] args) throws InterruptedException {
36 //单独的初始化线程,初始化分为2步,需要扣减两次
37 new Thread(new Runnable() {
38 @Override
39 public void run() {
40 SleepTools.ms(1);
41 System.out.println("Thread_"+Thread.currentThread().getId()
42 +" ready init work step 1st......");
43 latch.countDown();//每完成一步初始化工作,扣减一次
44 System.out.println("begin step 2nd.......");
45 SleepTools.ms(1);
46 System.out.println("Thread_"+Thread.currentThread().getId()
47 +" ready init work step 2nd......");
48 latch.countDown();//每完成一步初始化工作,扣减一次
49 }
50 }).start();
51 new Thread(new BusiThread()).start();
52 for(int i=0;i<=3;i++){
53 Thread thread = new Thread(new InitThread());
54 thread.start();
55 }
56
57 latch.await();
58 System.out.println("Main do ites work........");
59 }
60 }

运行结果可以看到,两个初始化线程先跑,当两个初始化线程跑完了,latch的计数器减为0,阻塞放开,主线程和业务线程继续往下运行。但是,在设计这部分算法的时候需要注意,有可能会出现计数器没有减为0,则线程一直阻塞,导致程序卡死。

countDownLatch一般使用在多线程并发之后需要对结果进行处理,而我们无法控制所有线程的执行时间,所以在这里加上阻塞,等到只有线程全部执行完。

2.CyclicBarrier

          cyclicBarrier从业务角度来说,和countDownLatch比较类似(具体对比后面会专门介绍),其作用类似一个屏障(英文中也是屏障的意思),阻隔线程直到全部到达屏障点,放开屏障。一般可以用在多线程统计的时候,

从代码角度来看,cyclicBarrier有两个构造方法,CyclicBarrier(int parties)和CyclicBarrier(int parties, Runnable barrierAction);

两个方法的区别是第一个只是记录了线程计数器的个数,而第二个不仅记录了计数器,当屏障放开时,会执行第二个参数线程的方法(第二个参数一般传入的是一个实现了Runnable的线程方法)

再来结合代码深入了解一下:

 1 public class UseCyclicBarrier {
2
3 private static CyclicBarrier barrier
4 = new CyclicBarrier(5,new CollectThread());
5
6 private static ConcurrentHashMap<String,Long> resultMap
7 = new ConcurrentHashMap<>();//存放子线程工作结果的容器
8
9 public static void main(String[] args) {
10 new Thread(){
11 @Override
12 public void run() {
13 long id = Thread.currentThread().getId();//线程本身的处理结果
14 resultMap.put(Thread.currentThread().getId()+"",id);
15 Random r = new Random();//随机决定工作线程的是否睡眠
16 try {
17 if(r.nextBoolean()) {
18 Thread.sleep(2000+id);
19 System.out.println("Thread_"+id+" ....do something ");
20 }
21 System.out.println(id+"....is await");
22 barrier.await();
23 Thread.sleep(1000+id);
24 System.out.println("Thread_"+id+" ....do its business ");
25 } catch (Exception e) {
26 e.printStackTrace();
27 }
28 }
29 }.start();
30 for(int i=0;i<=3;i++){
31 Thread thread = new Thread(new SubThread());
32 thread.start();
33 }
34 }
35
36 //负责屏障开放以后的工作
37 private static class CollectThread implements Runnable{
38
39 @Override
40 public void run() {
41 StringBuilder result = new StringBuilder();
42 for(Map.Entry<String,Long> workResult:resultMap.entrySet()){
43 result.append("["+workResult.getValue()+"]");
44 }
45 System.out.println(" the result = "+ result);
46 System.out.println("do other business........");
47 }
48 }
49
50 //工作线程
51 private static class SubThread implements Runnable{
52
53 @Override
54 public void run() {
55 long id = Thread.currentThread().getId();//线程本身的处理结果
56 resultMap.put(Thread.currentThread().getId()+"",id);
57 Random r = new Random();//随机决定工作线程的是否睡眠
58 try {
59 if(r.nextBoolean()) {
60 Thread.sleep(2000+id);
61 System.out.println("Thread_"+id+" ....do something ");
62 }
63 System.out.println(id+"....is await");
64 barrier.await();
65 Thread.sleep(1000+id);
66 System.out.println("Thread_"+id+" ....do its business ");
67 } catch (Exception e) {
68 e.printStackTrace();
69 }
70 }
71 }
72 }

输出结果:       11....is await
                            12....is await
                             Thread_10 ....do something
                     10....is await
Thread_13 ....do something
13....is await
Thread_14 ....do something
14....is await
 the result = [11][12][13][14][10]
do other business........
Thread_10 ....do its business
Thread_11 ....do its business
Thread_12 ....do its business
Thread_13 ....do its business
Thread_14 ....do its business

3.countDownLatch和cyclicBarrier区别

        (1)countDownlatch的计数器由调用countDown方法次数决定,每次调用计数器减一,可以在一个线程中调用多次,而cyclicBarrier计数器取决调用await方法的线程个数。

(2)  countDownLatch只能用一次,而cyclicBarrier可以循环使用。并且cyclicBarrier可以调用reset方法重置计数器,可以在线程故障重新启用线程调用。

(3)  二者在内部方法也有很多区别,具体有兴趣的可以去看看源码。

参考文章:http://ifeve.com/concurrency-cyclicbarrier/

          

并发编程常用工具类(一) countDownLatch和cyclicBarrier的使用对比的更多相关文章

  1. 并发编程常用工具类(二) SymaPhore实现线程池

    1.symaPhore简介 symaphore(信号量)用来控制同时访问某个资源的线程数量,一般用在并发流量控制.个人对它的理解相当于是接待室每次只能接待固定数量的人,当达到最高接待数的时候,其他人就 ...

  2. 【Java并发工具类】CountDownLatch和CyclicBarrier

    前言 下面介绍协调让多线程步调一致的两个工具类:CountDownLatch和CyclicBarrier. CountDownLatch和CyclicBarrier的用途介绍 CountDownLat ...

  3. 并发工具类:CountDownLatch、CyclicBarrier、Semaphore

    在多线程的场景下,有些并发流程需要人为来控制,在JDK的并发包里提供了几个并发工具类:CountDownLatch.CyclicBarrier.Semaphore. 一.CountDownLatch ...

  4. Java中的并发工具类:CountDownLatch、CyclicBarrier和Semaphore

    在java 1.5中,提供了一些非常有用的辅助类来帮助我们进行并发编程,比如CountDownLatch,CyclicBarrier和Semaphore,今天我们就来学习一下这三个辅助类的用法. 一. ...

  5. 多线程工具类:CountDownLatch、CyclicBarrier、Semaphore、LockSupport

    ◆CountDownLatch◆ 假如有一个任务想要往下执行,但必须要等到其他的任务执行完毕后才可以.比如你想要买套房子,但是呢你现在手上没有钱.你得等这个月工资发了.然后年终奖发了.然后朋友借你得钱 ...

  6. 并发编程JUC系列AQS(CountDownLatch、CyclicBarrier、Semaphore)

    一.CountDownLatch package com.jonychen.test; import java.util.concurrent.CountDownLatch; import java. ...

  7. 多线程进阶之并发工具类:CountDownLatch、CyclicBarrier

    并发工具类在java.util.concurrent包下.常用的有CountDownLatch.CyclicBarrier,用它们可以控制并发流程. 1.CountDownLatch探究: 主要用到其 ...

  8. java并发编程 - Exexctors 工具类

    Executors 类提供了一系列静态工厂方法用于创建各种线程池. newFixedThreadPool 创建固定大小的线程池.每次提交一个任务就创建一个线程,直到线程达到线程池的最大大小.线程池的大 ...

  9. 并发编程--Concurrent-工具类介绍

    并发编程--Concurrent-工具类介绍 并发编程--Concurrent-工具类介绍 CountDownLatch CylicBarrier Semaphore Condition 对象监视器下 ...

随机推荐

  1. 第六章 Sleuth--链路追踪

    修整了2天,我们继续接着上篇 第五章 Gateway–服务网关 继续来讲SpringCloud Alibaba全家桶中的 Sleuth 链路追踪 组件 喜欢记得点关注哦 6.1 链路追踪介绍 在大型系 ...

  2. 不一样的资产安全 3D 可视化平台

    前言   数字经济时代,应用好数据是企业数字化转型的关键,基于前沿科学技术进行数据的有效管控,更是对数字增值服务的新趋势.近年来,整个安全行业对资产管理的重视程度正在提高.据IDC发布的相关数据显示, ...

  3. 精尽Spring MVC源码分析 - LocaleResolver 组件

    该系列文档是本人在学习 Spring MVC 的源码过程中总结下来的,可能对读者不太友好,请结合我的源码注释 Spring MVC 源码分析 GitHub 地址 进行阅读 Spring 版本:5.2. ...

  4. CCNP之静态路由实验报告

                   静态路由实验报告 一.实验要求: 1.内网IP基于172.16.0.0/16自行子网划分 2.除了R2--R4路由器各有两个环回接口 3.R1下的PC自动获取IP地址 4 ...

  5. Quatz JobListener和TriggerListener

    myJob:triggerFired... vetoJobExecution class coder.rdf.mybatis.study.JobTest:jobToBeExecuted... test ...

  6. [leetcode]Q133Clone Graph

    克隆图记住:一个map一个queue,照葫芦画瓢BFS 找到一个节点就画一个对应的新的,用map对应,然后添加邻居,记录邻居到queue public UndirectedGraphNode clon ...

  7. 【陪你系列】5 千字长文+ 30 张图解 | 陪你手撕 STL 空间配置器源码

    大家好,我是小贺. 点赞再看,养成习惯 文章每周持续更新,可以微信搜索「herongwei」第一时间阅读和催更,本文 GitHub https://github.com/rongweihe/MoreT ...

  8. 基于E-PUCK 2.0多智能体自主协同 高频投影定位系统

    群体智能机器人是一种国际前沿的人工智能研究项目,由多个小型机器人组成的集群式解决系统,灵感源于蚂蚁.蜜蜂.鱼等群体生物,在没有统一领导的情况下,也能合作执行大量复杂的任务,比如组建一个图形,再在此基础 ...

  9. RPC框架从0到10

    RPC(Remote Procedure Call) 从单机走向分布式,产生了很多分布式的通信方式 最古老也是最有效,并且永不过时的,TCP/UDP的二进制传输,事实上所有的通信方式归根结底都是TCP ...

  10. 十、scala、spark集群搭建

    spark集群搭建: 1.上传scala-2.10.6.tgz到master 2.解压scala-2.10.6.tgz 3.配置环境变量 export SCALA_HOME=/mnt/scala-2. ...