CountDownLatch:

  可以让一个线程等待其他线程完成了各自的工作之后再执行。比如说一个切菜,一个人切肉,都准备完毕之后才能炒肉。

构造方法:

public CountDownLatch(int count)  count等待的线程数量

关键API:

countDown()   分线程执行完减少计数

await()        主线程等待调用

使用:

package com.nijunyang.concurrent;

import java.util.concurrent.CountDownLatch;

/**
* Description:
* Created by nijunyang on 2020/5/16 13:53
*/
public class CountDownLatchTest{ private CountDownLatch countDownLatch; public CountDownLatchTest(CountDownLatch countDownLatch) {
this.countDownLatch = countDownLatch;
} public static void main(String[] args) throws InterruptedException { CountDownLatch countDownLatch = new CountDownLatch(2);
CountDownLatchTest countDownLatchTest = new CountDownLatchTest(countDownLatch); new Thread(()-> {
try {
countDownLatchTest.method1();
} catch (InterruptedException e) {
e.printStackTrace();
}
},"线程1").start();
new Thread(()-> {
try {
countDownLatchTest.method2();
} catch (InterruptedException e) {
e.printStackTrace();
}
},"线程2").start(); System.out.println("等待食材准备完毕...");
countDownLatch.await();
System.out.println("炒肉..."); // System.out.println("------第二次使用-----");
// new Thread(()-> {
// try {
// countDownLatchTest.method1();
// } catch (InterruptedException e) {
// e.printStackTrace();
// }
// },"线程1").start();
// new Thread(()-> {
// try {
// countDownLatchTest.method2();
// } catch (InterruptedException e) {
// e.printStackTrace();
// }
// },"线程2").start();
//
// System.out.println("等待食材准备完毕...");
// countDownLatch.await();
// System.out.println("炒肉..."); } private void method1() throws InterruptedException {
Thread.sleep(5000L);
System.out.println("切菜完毕...");
countDownLatch.countDown();
} private void method2() throws InterruptedException {
Thread.sleep(10000L);
System.out.println("切肉完毕...");
countDownLatch.countDown();
}
}

原理解析

1.从构造方法进去我们可以看到又是一个熟悉的Sync内部类继承了AbstractQueuedSynchronizer,入参的数量被赋值到AbstractQueuedSynchronizer的state字段。

2.await方法会去判断state是否等于0,如果不等于0,说明其他线程还没有执行完毕。就会执行doAcquireSharedInterruptibly这个方法,将当前这个调用await方法的线程入队阻塞。

(调用链:await()-sync.acquireSharedInterruptibly-sync.tryAcquireShared-doAcquireSharedInterruptibly)

3.countDown方法,每调一次就会将state的值减1,当扣减到0的时候去唤醒上面等待的主线程执行(调用链:countDown-sync.releaseShared-sync.tryReleaseShared-doReleaseShared(减到0才会执行这方法))

CyclicBarrier

篱栅,顾名思义有拦截作用。它可以让一组线程到达栅栏时被阻塞,直到最后一个线程到达,才放行通过。比如玩LOL,需要等待所有玩家进度条100%了,才能进入游戏

构造方法:

CyclicBarrier(int parties)      parties:阻塞的线程数量

CyclicBarrier(int parties, Runnable barrierAction)  parties:阻塞的线程数量  barrierAction:当最后一个线程到达是先执行这个任务,再去执行后面的流程。

关键API:

await()  到达栅栏点等待。调用次数要和入参数量一致,否则会一致阻塞的等待。

使用

package com.nijunyang.concurrent;

import java.util.concurrent.CyclicBarrier;

/**
* @author: create by nijunyang
* @date:2019/9/5
*/
public class CyclicBarrierTest implements Runnable{
private CyclicBarrier cyclicBarrier;
public CyclicBarrierTest(CyclicBarrier cyclicBarrier) {
this.cyclicBarrier = cyclicBarrier;
}
public void run() {
try {
System.out.println(Thread.currentThread().getName() + "进度条100%... ");
cyclicBarrier.await();
} catch (Exception e) {
e.printStackTrace();
}
}
public static void main(String[] args) throws Exception {
CyclicBarrier cyclicBarrier = new CyclicBarrier(11, new Runnable() {//11个是因为还有一个主线程也在等待
public void run() {
System.out.println("所有人进度条100%,准备开始游戏");
}
});
for (int i = 0; i < 10; i++) {
new Thread(new CyclicBarrierTest(cyclicBarrier), "线程" + i).start();
}
cyclicBarrier.await();
Thread.sleep(300);
System.out.println("开始游戏....");
} }

原理解析

1.默认每个CyclicBarrier对象有一把锁ReentrantLock和Condition

2.将构造方法的入参数量赋值到count字段中。后续都是在count字段上面进行操作。

3.await的调用会将count的数量-1,如果扣减到0.则会先执行构造方法传入的任务(如果传了),并且重置计数器刷新栅栏,将许可数据重新赋值给count字段(可以重复使用),唤醒条件等待的线程

4.如果扣减完成之后还没有到0.说明还有线程没有到达栅栏点。则进入条件队列阻塞等到,等到最后一个到达时候,才被唤醒

 两者比较

CountDownLatch和CyclicBarrier,最终实现效果看起来都差不多,都是等待分支线程执行完毕,再往下执行。然后CyclicBarrier这个可以重复使用,因为会去刷新count的数量。CountDownLatch不会重新刷新state字段的值。当第二次await执行的时候一看state是0就直接放行了,所以一个CountDownLatch对象只能使用一次。

原理上CountDownLatch是阻塞主线程,分支线线程执行完毕将state扣减到0了之后唤醒主线程去执行,CyclicBarrier则是所有线程到达栅栏点都会阻塞等待。直到后一个到达才唤醒所有的阻塞线程。

JUC(3)---CountDownLatch、CyclicBarrier和AQS的更多相关文章

  1. CountDownLatch/CyclicBarrier/Semaphore 使用过吗?

    CountDownLatch/CyclicBarrier/Semaphore 使用过吗?下面详细介绍用法: 一,(等待多线程完成的)CountDownLatch  背景; countDownLatch ...

  2. 并发包下常见的同步工具类详解(CountDownLatch,CyclicBarrier,Semaphore)

    目录 1. 前言 2. 闭锁CountDownLatch 2.1 CountDownLatch功能简介 2.2 使用CountDownLatch 2.3 CountDownLatch原理浅析 3.循环 ...

  3. CountDownLatch CyclicBarrier和 Semaphore

    CountDownLatch CyclicBarrier和 Semaphore 原理 基于AQS实现. 让需要的暂时阻塞的线程,进入一个死循环里面,得到某个条件后再退出循环,以此实现阻塞当前线程的效果 ...

  4. java 并发工具类CountDownLatch & CyclicBarrier

    一起在java1.5被引入的并发工具类还有CountDownLatch.CyclicBarrier.Semaphore.ConcurrentHashMap和BlockingQueue,它们都存在于ja ...

  5. 【JUC】CountDownLatch

    因为在调用端的异步中,需要调用其他多个服务获取数据再汇总结果返回,所以用到了CountDownLatch CountDownLatch的概念 CountDownLatch是一个同步工具类,用来协调多个 ...

  6. Java并发编程工具类 CountDownLatch CyclicBarrier Semaphore使用Demo

    Java并发编程工具类 CountDownLatch CyclicBarrier Semaphore使用Demo CountDownLatch countDownLatch这个类使一个线程等待其他线程 ...

  7. 并发包下常见的同步工具类(CountDownLatch,CyclicBarrier,Semaphore)

    在实际开发中,碰上CPU密集且执行时间非常耗时的任务,通常我们会选择将该任务进行分割,以多线程方式同时执行若干个子任务,等这些子任务都执行完后再将所得的结果进行合并.这正是著名的map-reduce思 ...

  8. 高并发第十单:J.U.C AQS(AbstractQueuedSynchronizer) 组件:CountDownLatch. CyclicBarrier .Semaphore

    这里有一篇介绍AQS的文章 非常好: Java并发之AQS详解 AQS全名:AbstractQueuedSynchronizer,是并发容器J.U.C(java.lang.concurrent)下lo ...

  9. JUC之CountDownLatch和CyclicBarrier的区别 (转)

    CountDownLatch和CyclicBarrier的功能看起来很相似,不易区分,有一种谜之的神秘.本文将通过通俗的例子并结合代码讲解两者的使用方法和区别. CountDownLatch和Cycl ...

随机推荐

  1. ios快捷指令编程尝试

    最近,,,啊好几个月了,发现这个ios的快捷指令很好玩 原生就提供了不少功能 用来练习编程思维是十分有用啊...) 其次呢,还可以使用外接的功能对原有的功能进行拓展,比如api借口啊,ssh执行程序啊 ...

  2. api接口安全

    API接口安全 在做app开发中,如何保证api的接口安全,不被其他app去调用? 接口安全的措施很多,今天记录一个常用的措施 签名: 前台想要调用接口,需要使用几个参数生成签名: 时间戳:当前时间 ...

  3. MVC-路由扩展-限制浏览器

    根据路由原理,MVC每次都会走获取路由上下文数据. 自定义Route 调用,以及完善其他代码 运行结果,当在谷浏览器执行时:

  4. CentOS 7 + Win 双系统的安装遇到的重要问题

    前言:对于刚学linux的朋友们,多多小小因为各种原因需要装双系统,亦或者爱好使然.多数是问题解决,第一次装系统者不推荐看-. 那么现在内德在此就说说在本本上装双系统会遇到的问题及其解决方法. 环境准 ...

  5. printf 参数检查 __attribute__((format(printf, 1, 2)))

    With GCC, I can specify __attribute__((format(printf, 1, 2))) , telling the compiler that this funct ...

  6. Zabbix3.0安装部署最佳实践

    Zabbix介绍 1.1zabbix 简介 Zabbix 是一个高度集成的网络监控解决方案,可以提供企业级的开源分布式监控解决方案,由一个国外的团队持续维护更新,软件可以自由下载使用,运作团队靠提供收 ...

  7. eclipse自动补全导致变量会跟上String后缀的问题解决

    https://blog.csdn.net/feinifi/article/details/103665860

  8. while循环脚本

    [root@oldboy ~]# (while :;do date;sleep 5;done)& fg ctrl c退出 fg ( while :; do date; sleep 5; don ...

  9. 杭电60题--part 1 HDU1003 Max Sum(DP 动态规划)

    最近想学DP,锻炼思维,记录一下自己踩到的坑,来写一波详细的结题报告,持续更新. 题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1003 Problem ...

  10. P3807【模板】卢卡斯定理

    题解大部分都是递归实现的,给出一种非递归的形式 话说上课老师讲的时候没给代码,然后自己些就写成了这样 对于质数\(p\)给出卢卡斯定理: \[\tbinom{n}{m}=\tbinom{n \bmod ...