老司机发车了,CountDownLatch:等与不等都在你

news2024/7/6 19:06:06

哈喽大家好,我是阿Q。

前几天我们把 ReentrantLock的原理 进行了详细的讲解,不熟悉的同学可以翻看前文,今天我们介绍另一种基于 AQS 的同步工具——CountDownLatch。

CountDownLatch 被称为倒计时器,也叫闭锁,是 juc 包下的工具类,同时也是共享锁的一种实现。它的作用是可以让一个或多个线程等待,直到所有线程的任务都执行完之后再继续往下执行。

举个简单的例子:阿Q高中时期都是乘坐大巴往返于县城与农村,那时的司机为了利益的最大化,会在汽车满员的情况下才会发车。

如果我们把乘客去车站乘车比作一个一个的线程,那 CountDownLatch 做的事就是等大家到齐之前的等待工作。

我们从源码的角度来分析下它的工作原理

①谁来决定公交车上的座位数?

公交车上的座位数是由汽车制造商决定的,在 CountDownLatch 中也会存在这样一个值 count,用来表示需要等待的线程个数

count 值是在 CountDownLatch 的构造函数中进行初始化的

public CountDownLatch(int count) {
	if (count < 0) throw new IllegalArgumentException("count < 0");
	this.sync = new Sync(count);
}

Sync(int count) {
 //设置 AQS 中的 state 为 count 值
	setState(count);
}
复制代码

计数值 count 是一次性的,当它的值减为0后就不会再变化了,这也是其存在的不足之处。

②谁来确定乘客全部到齐?

在汽车发车前检票员会对车上的乘客数量进行清点,如果满员了就会通知司机开车。

当然也可以采用这种方法:在得知车座位数的前提下,每上来一位乘客,座位数进行减一操作。CountDownLatch 就是采用的上述方法,它的 countDown() 方法会对 state 的值执行减1操作。

让我们从源码的角度来认识一下该方法。

public void countDown() {
 //释放共享锁
	sync.releaseShared(1);
}

public final boolean releaseShared(int arg) {
	if (tryReleaseShared(arg)) {
		doReleaseShared();
		return true;
	}
	return false;
}
复制代码

先尝试释放锁,如果返回 true,则执行释放操作,反之不执行。我们分析下上边的两个方法

protected boolean tryReleaseShared(int releases) {
	for (;;) {
  //获取当前等待的线程数量
		int c = getState();
  //等待线程数为0,表示没有等待线程,故不需要释放锁资源
		if (c == 0)
			return false;
  //执行减1操作
		int nextc = c-1;
  //自旋+CAS将state的属性值-1
		if (compareAndSetState(c, nextc))
			return nextc == 0;
	}
}
复制代码

最后一步中,如果减一之后为0,则说明没有其它线程等待,需要执行释放锁操作,返回 true,反之不需要。

在开始分析 doReleaseShared() 之前,我们先来补全一下 AQS 中 waitStatus 的状态说明

  • 初始化状态:0,表示当前节点在同步队列中,等待获取锁;
  • CANCELLED:1,表示当前节点取消获取锁;
  • SIGNAL:-1,表示后续节点等待当前节点唤醒;
  • CONDITION:-2,表示当前线程正在条件等待队列中;
  • PROPAGATE:-3,共享模式,前置节点唤醒后续节点后,唤醒操作无条件传播下去;
/**
 * 释放锁:唤醒后续节点
 */
private void doReleaseShared() {
	for (;;) {
		Node h = head;
  //不是null 且不为尾节点,因为尾节点没有后续节点需要唤醒了
		if (h != null && h != tail) {
			int ws = h.waitStatus;
   //只有状态为 -1 才可以唤醒后续节点
			if (ws == Node.SIGNAL) {
				//将waitStatus设置为0失败会继续循环
				if (!compareAndSetWaitStatus(h, Node.SIGNAL, 0))
					continue;
				unparkSuccessor(h);
			}
			//将waitStatus设置为PROPAGATE失败会继续循环
			else if (ws == 0 &&
					 !compareAndSetWaitStatus(h, 0, Node.PROPAGATE))
				continue;                
		}
		if (h == head)                   
			break;
	}
}
复制代码

unparkSuccessor() 方法用于唤醒 AQS 中被挂起的线程,在ReentrantLock的原理中讲过了,此处不再赘述。

小结:当线程使用 countDown() 方法时,其实是使用了 tryReleaseShared() 方法以 CAS 的操作来减少 state ,直至 state 为 0 ,进而释放锁资源,唤醒后续节点。

③谁来发车?

肯定是司机来发车呀,那我们的 CountDownLatch 是如何实现的呢?

CountDownLatch 中的 await() 方法,就是等待线程的总开关,当发现 state 的值为0时会释放所有的等待线程,发车了。

我们从源码角度来看下它是如何工作的

public void await() throws InterruptedException {
	sync.acquireSharedInterruptibly(1);
}

public final void acquireSharedInterruptibly(int arg)
		throws InterruptedException {
 //如果线程中断了,直接抛出中断异常
	if (Thread.interrupted())
		throw new InterruptedException();
 //如果小于0,代表 state 不为0,即还有任务未执行完毕,会执行获取共享锁的操作
	if (tryAcquireShared(arg) < 0)
		doAcquireSharedInterruptibly(arg);
}

protected int tryAcquireShared(int acquires) {
	return (getState() == 0) ? 1 : -1;
}
复制代码

我们来看看它到底是如何获取共享锁的

private void doAcquireSharedInterruptibly(int arg)
	throws InterruptedException {
	//将当前线程封装成node放到队尾
	final Node node = addWaiter(Node.SHARED);
	boolean failed = true;
	try {
		for (;;) {
			final Node p = node.predecessor();
			if (p == head) {
				int r = tryAcquireShared(arg);
				//state为0,表示此时等待线程全部执行完毕,r为1。
				if (r >= 0) {
					setHeadAndPropagate(node, r);
					p.next = null;
					failed = false;
					return;
				}
			}
			//从当前node节点向前寻找有效节点,并保证有效节点的waitStatus状态为-1
			if (shouldParkAfterFailedAcquire(p, node) &&
				//挂起线程
				parkAndCheckInterrupt())
				//在拿锁的期间,如果被中断了,那么会抛出异常,取消拿锁
				throw new InterruptedException();
		}
	} finally {
		if (failed)
			//将当前节点设置为失效节点,并挂到最近的有效节点后边,上文中有图解
			cancelAcquire(node);
	}
}
复制代码

其中最重要的就是 setHeadAndPropagate() 方法

private void setHeadAndPropagate(Node node, int propagate) {
	Node h = head; 
	//将当前node设置为head,并将node的线程置为空
	setHead(node);
	if (propagate > 0 || h == null || h.waitStatus < 0 ||
		(h = head) == null || h.waitStatus < 0) {
		Node s = node.next;
		if (s == null || s.isShared())
			//释放锁:唤醒后续节点
			doReleaseShared();
	}
}
复制代码

小结:当线程使用 await() 方法时会将当前线程封装成 node 加入AQS 队列中,如果发现 state 不为0,说明还有任务未执行完成,继续阻塞;如果 state 为0,会释放掉所有的等待线程,执行 await() 之后的数据。

流程图了解一下

理论讲完了,那我们用代码来演示下上边的例子

public static void main(String[] args) throws InterruptedException {
	int count = 10;
	//设置线程池并发数
	ExecutorService executorService = Executors.newFixedThreadPool(count);
	//假设大巴可以拉十个乘客,初始化state
	CountDownLatch countDownLatch = new CountDownLatch(count);
	for (int i = 0; i < count; i++) {
		final int num = i;
		executorService.execute(()->{
			try {
				Thread.sleep((long) (new Random().nextDouble() * 3000) + 1000);
				System.out.println("乘客坐在了"+ (num +1) + "号座位上");
			} catch (InterruptedException exception) {
				exception.printStackTrace();
			}finally {
				countDownLatch.countDown();
			}
		});
	}
	System.out.println("司机等待乘客上车");
	countDownLatch.await();
	System.out.println("发车了");
	executorService.shutdown();
}
复制代码

执行结果如下:

细心地同学肯定会问了:如果遇上刮风下雨,来坐车的人少了,那已经上车的乘客岂不是回不了家了?

当然不是了,大巴其实也是有时间观念的,即使车上的乘客不满员到了一定的时间司机也会发车的,另外还会在路上顺道稍几个人上车。那我们的 CountDownLatch 是如何实现的呢?

CountDownLatch 还提供了一个 await(long timeout, TimeUnit unit)方法,在一定的时间间隔内会阻塞当前线程,等待 count 个线程执行任务,一旦超出了等待时间,便会继续往下执行。

我们将上边的countDownLatch.await();替换为countDownLatch.await(3, TimeUnit.SECONDS);,执行结果如下所示

上文中的例子是 CountDownLatch 的其中一种用法,即主线程等待其他线程执行完毕之后再执行。它还有另一种用法,即实现多个线程开始执行任务的最大并行性,类似发令枪响前,运动员统一在起跑线就位的场景。

public static void main(String[] args) throws InterruptedException {
	//设置线程池并发数
	ExecutorService executorService = Executors.newFixedThreadPool(10);
	CountDownLatch countDownLatch = new CountDownLatch(1);
	//一组有6名运动员
	for (int i = 0; i < 6; i++) {
		final int num = i;
		executorService.execute(()->{
			try {
				System.out.println("运动员"+ (num+1) +"等待发令枪响");
				countDownLatch.await();
				System.out.println("运动员"+ (num+1) +"开始起跑");
			} catch (InterruptedException exception) {
				exception.printStackTrace();
			}
		});
	}
	Thread.sleep(3000);
	countDownLatch.countDown();
	System.out.println("发令枪响");
	executorService.shutdown();
}
复制代码

执行结果如下

说了这么多,都是样例?你有没有在项目中应用过呢?

回答当然是“Yes”了,之前的运营端有个统计页面,要求统计用户新增数量、订单数量、商品交易总额等多张表的指标值,为了提高执行速率,我就启用了多个子线程分别去统计,用 CountDownLatch 来等待它们的统计结果。

今天的内容到这里就结束了,希望对大家有所帮助,我们下期再见。写文不易,希望大家可以一键三连:点赞、转发、在看。

微信搜索公众号《Java水解蛋白》:面试题、pdf、简历、资料、3113有超赞的粉丝福利,也可以来技术群来讨论问题呦。

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

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

相关文章

[论文精读|顶刊论文]Relational Triple Extraction: One Step is Enough

2022.5.11 &#xff5c;IJCAI-2022&#xff5c;华中科技大学&#xff5c;2022年SOTA&#xff5c; 原文链接 Relational Triple Extraction: One Step is Enough 过去的步骤&#xff1a; 寻找头尾实体的边界位置&#xff08;实体识别&#xff09;将特定令牌串联成三元组&…

[附源码]Python计算机毕业设计Django区域医疗服务监管可视化系统

项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等等。 环境需要 1.运行环境&#xff1a;最好是python3.7.7&#xff0c;…

分享107个小清新,总有一款适合您

PPT链接&#xff1a;https://pan.baidu.com/s/1WqaR_29avEgq46iTSLKfmw?pwd5r81 提取码&#xff1a;5r81 源码下载链接&#xff1a;ppt.rar - 蓝奏云 采集的参数 page_count 1 # 每个栏目开始业务content"text/html; charsetgb2312"base_url "https://sc…

Python可视化招聘信息聚合系统 (附源码)!

前言 基于数据技术的互联网行业招聘信息聚合系统&#xff0c;本系统以Python为核心&#xff0c;依托web展示&#xff0c;所有功能在网页就可以完成操作&#xff0c;爬虫、分析、可视化、互动独立成模块&#xff0c;互通有无。 依托python的丰富库实现&#xff0c;爬虫使用Req…

详解设计模式:备忘录模式

详解设计模式&#xff1a;备忘录模式 备忘录模式&#xff08;Memento Pattern&#xff09;也被称为快照模式&#xff08;Snapshot Pattern&#xff09;、Token 模式&#xff08;Token Pattern&#xff09;&#xff0c;是在 GoF 23 种设计模式中定义了的行为型模式。 备忘录模式…

阿里云存储解决方案,助力轻舟智航“将无人驾驶带进现实”

轻舟智航介绍 轻舟智航是一家以“将无人驾驶带进现实”为使命的自动驾驶通用解决方案公司&#xff0c;依赖双擎战略&#xff0c;一方面主张以高性价比的前装量产方案&#xff0c;致力于打造L4级体验的城市高速NOA方案&#xff0c;满足不同客户不同等级的自动驾驶量产需求。另一…

Lottie 动画导出为 GIF/MP4 以及与 QML 集成演示

获取 Lottie 动画文件 lottiefiles 是一个很好的网站, 从上面可以下载到别人分享的 lottie 动画文件. 我们可以下载到多种格式, 下面分别讲解每个格式的下载和适用情景. 下载 JSON 源文件 这是体积最小的格式, 一般在 10kb ~ 100kb 之间. 考虑到 lottiefiles 的服务器在国外…

锂热电池检测设备 你一定没见过这种检测方式!

项目需求 用户希望纳米Namisoft帮他们设计开发一款系统&#xff0c;要求系统软件安装在PC控制装置上&#xff0c;系统通过使用USB、RS232、LAN通讯接口实现对锂电池测试过程中所用到的仪器&#xff08;内阻测试仪、扫码枪、触摸显示器和电源模块等&#xff09;进行软件控制&…

浸没式冷却-散热技术新趋势,一起学Flotherm电子元器件散热仿真

作者&#xff1a;Billy&#xff0c;仿真秀专栏作者 随着电子元器件功率的上升&#xff0c;散热成为技术发展的瓶颈之一。单纯的风冷在一些情况下无法满足散热需求&#xff0c;直接式液冷和间接式液冷因其可以提供更大量级的对流换热系数&#xff0c;带走更多的热量&#xff0c…

基于智能优化算法PSO/GWO/AFO+柔性车间生产调度(Matlab代码实现)

目录 1 柔性车间生产调度 2 运行结果 3 参考文献 4 Matlab代码实现 1 柔性车间生产调度 随着经济全球化的不断加深和市场竞争的日益严峻,传统的单一车间制造模式已经无法满足我国制造业的生产需求,分布式生产制造模式已经成为企业提高生产竞争力的重要手段。由于不同工厂之…

[附源码]计算机毕业设计校友社交系统Springboot程序

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

目标检测算法——3D公共数据集汇总 2(附下载链接)

>>>深度学习Tricks&#xff0c;第一时间送达<<< &#x1f384;&#x1f384;近期&#xff0c;小海带在空闲之余&#xff0c;收集整理了一批3D公共数据集供大家参考。 整理不易&#xff0c;小伙伴们记得一键三连喔&#xff01;&#xff01;&#xff01;&…

微服务自动化【Docker-Compose】

目录 1. docker-compose 2. docker-compose安装与配置 3. docker-compose.yml 配置文件基本介绍 3.1 version: 指定 docker-compose.yml 文件的写法格式 3.2 services&#xff1a;多个容器集合 4. docker-compose 基本指令 5. docker-compose 网络 5.1 指定网络模式 …

不懂业务不清楚指标?这40套可视化大屏模板,让你突破职场天花板

报表可以说是中国职场的一大特色&#xff0c;不少职场人需要每天做各种报表给领导或者业务决策者看&#xff0c;为此甚至诞生了不少的“表哥表姐”。但很多人在做报表的时候其实并不懂业务&#xff0c;需要找业务确定业务指标才做的下去。 今天我就分享40多个报表模板&#xf…

Spark 3.0 - 10.Ml 常用 Sample 采样方法

目录 一.引言 二.数据准备 三.随机采样 Sample 四.按权重拆分 randomSplit 五.分层采样 sampleByKey 六.总结 一.引言 使用 Spark 进行机器学习、数据分析等项目时&#xff0c;常常需要对数据进行采样&#xff0c;下面介绍三种最常用的采样方法&#xff1a; A.随机采样:…

Vue3 学习笔记 —— 自动导入 Vue3 APIs、v-model

目录 1. 自动导入 Vue3 APIs —— unplugin-auto-import/vite 2. v-model 2.1 相较于 Vue2&#xff0c;Vue3 做出了哪些变化&#xff1f; 2.2 绑定一个 v-model 2.2.1 父组件 2.2.2 子组件 2.3 绑定多个 v-model 2.3.1 父组件 2.3.2 子组件 2.4 v-model 中的自定义修…

Android Material Design之ShapeableImageView(十三)

效果图 资源引入 implementation com.google.android.material:material:1.4.0属性 属性描述android:id控件idandroid:layout_width控件长度android:layout_height控件高度app:shapeAppearance控件外观样式app:strokeWidth画笔粗度app:strokeColor画笔颜色android:src图像资源…

MySQL逻辑架构

逻辑架构剖析 服务器处理客户端请求 数据库查询请求流程&#xff1a; 连接层 系统&#xff08;客户端&#xff09;访问 MySQL 服务器前&#xff0c;做的第一件事就是建立 TCP 连接。 经过三次握手建立连接成功后&#xff0c; MySQL 服务器对 TCP 传输过来的账号密码做身份认…

了解Wi-fi频段概念

前言 信道带宽&#xff0c;应该了解wi-fi频段&#xff0c;这样才能分析有多少信道带宽可用&#xff0c;以及如何在没有任何干扰&#xff08;失真&#xff09;的情况下有效地使用它。 2.4GHz和5GHz频段可用于wi-fi。 2.4 GHz Wi-Fi频段&#xff1a;在2.4 GHz频段&#xff0c;…

通过Shell脚本自动安装HiveJDBC测试提供CDH5网盘地址

〇、参考地址 1、Linux下编写脚本自动安装hive https://blog.csdn.net/weixin_44911081/article/details/121227024?ops_request_misc%257B%2522request%255Fid%2522%253A%2522163695916016780269859534%2522%252C%2522scm%2522%253A%252220140713.130102334.pc%255Fblog.%252…