【并发编程】CountDownLatch详解与原理

📫作者简介:小明Java问道之路,2022年度博客之星全国TOP3,专注于后端、中间件、计算机底层、架构设计演进与稳定性建设优化,文章内容兼具广度、深度、大厂技术方案,对待技术喜欢推理加验证,就职于知名金融公司后端高级工程师。

🏆 2022博客之星TOP3 | CSDN博客专家 | 后端领域优质创作者 | CSDN内容合伙人

🏆 InfoQ(极客邦)签约作者、阿里云专家 | 签约博主、51CTO专家 | TOP红人、华为云享专家

🔥如果此文还不错的话,还请👍关注、点赞、收藏三连支持👍一下博主~

本文目录

本文导读

一、什么是CountDownLatch

二、CountDownLatch实现原理

1、CountDownLatch数据结构

2、countDown() 方法

3、await() 方法

总结

本文导读

本文讲解什么是CountDownLatch,CountDownLatch的特点以及使用场景,对CountDownLatch的数据结构与countDown()方法、await()方法源码进行剖析。

一、什么是CountDownLatch

CountDownLatch可以使线程顺序执行

CountDownLatch 类在创建实例的时候,需在构造函数中传入倒数次数,然后由需要等待的线程去调用 await 方法开始等待,而每一次其他线程调用了 countDown 方法之后,计数便会减 1,直到减为 0 时,之前等待的线程便会继续运行。

CountDownLatch 不能重用,可以考虑使用 CyclicBarrier 或者创建一个新的 CountDownLatch 实例。

1、CountDownLatch可以让一个线程等待其他多个线程都执行完毕,再继续自己的工作
ini 复制代码
public static void main(String[] args) throws InterruptedException {
CountDownLatch latch = new CountDownLatch(5);
ExecutorService service = Executors.newFixedThreadPool(5);
for (int i = 0; i < 5; i++) {
  final int no = i + 1;
  Runnable runnable = new Runnable() {
  	@Override
  	public void run() {
  		latch.countDown();
  	}
  };
  service.submit(runnable);
  	}

latch.await();
System.out.println("所有线程都执行完");
          }

2、多个线程等待某一个线程的信号,同时开始执行

ini 复制代码
 public static void main(String[] args) throws InterruptedException {
CountDownLatch countDownLatch = new CountDownLatch(1);
ExecutorService service = Executors.newFixedThreadPool(5);
for (int i = 0; i < 5; i++) {
  final int no = i + 1;
  Runnable runnable = new Runnable() {
  	@Override
  	public void run() {
  		countDownLatch.await();
  	}
  };
  service.submit(runnable);
 }

System.out.println("执行开始!");
  countDownLatch.countDown();
    }

二、CountDownLatch实现原理

1、CountDownLatch数据结构

CountDownLatch是利用AQS共享锁机制的同步器Sync来实现的

java 复制代码
public class CountDownLatch {
// 继承AQS来实现同步器,
private static final class Sync extends AbstractQueuedSynchronizer {

  // 通过state控制count
    Sync(int count) {
        setState(count);
    }

  // 重写AQS的tryAcquireShared,通过获取共享锁的方式实现等待
    protected int tryAcquireShared(int acquires) {
  	// 当state不为0时一直阻塞
        return (getState() == 0) ? 1 : -1;
    }

  // 释放共享锁
    protected boolean tryReleaseShared(int releases) {
        for (;;) {
            int c = getState();
  		
  		// 如果state为0,则不能再使用,
            if (c == 0) return false;
  		// state减1
            int nextc = c-1;
  		// 为0,释放共享锁,唤醒等待线程
            if (compareAndSetState(c, nextc)) return nextc == 0;
        }
    }
}

// 同步器变量
private final Sync sync;
// 创建同步器,设置AQS的state变量为1
public CountDownLatch(int count) {
    if (count < 0) throw new IllegalArgumentException("count < 0");
    this.sync = new Sync(count);
}
}

2、countDown() 方法

csharp 复制代码
// 直接调用AQS的releaseShared,释放共享锁,从而回调tryAcquireShared
public void countDown() {
sync.releaseShared(1);
}

3、await() 方法

csharp 复制代码
public void await() throws InterruptedException {
// 调用AQSacquireSharedInterruptibly,释放共享锁
sync.acquireSharedInterruptibly(1);
 }

总结

本文讲解什么是CountDownLatch,CountDownLatch的特点以及使用场景,对CountDownLatch的数据结构与countDown()方法、await()方法源码进行剖析。

相关推荐
bobz96510 分钟前
动态规划
后端
stark张宇26 分钟前
VMware 虚拟机装 Linux Centos 7.9 保姆级教程(附资源包)
linux·后端
亚力山大抵1 小时前
实验六-使用PyMySQL数据存储的Flask登录系统-实验七-集成Flask-SocketIO的实时通信系统
后端·python·flask
超级小忍1 小时前
Spring Boot 中常用的工具类库及其使用示例(完整版)
spring boot·后端
CHENWENFEIc2 小时前
SpringBoot论坛系统安全测试实战报告
spring boot·后端·程序人生·spring·系统安全·安全测试
重庆小透明3 小时前
力扣刷题记录【1】146.LRU缓存
java·后端·学习·算法·leetcode·缓存
博观而约取3 小时前
Django 数据迁移全解析:makemigrations & migrate 常见错误与解决方案
后端·python·django
寻月隐君4 小时前
Rust 异步编程实践:从 Tokio 基础到阻塞任务处理模式
后端·rust·github
GO兔4 小时前
开篇:GORM入门——Go语言的ORM王者
开发语言·后端·golang·go
Sincerelyplz4 小时前
【Temproal】快速了解Temproal的核心概念以及使用
笔记·后端·开源