JUC包:CountDownLatch源码+实例讲解

news2024/11/26 22:26:40

1 缘起

有一次听到同事谈及AQS时,我有很多点懵,
只知道入队和出队,CLH(Craig,Landin and Hagersten)锁,并不了解AQS的应用,
同时结合之前遇到的多线程等待应用场景,发现CountDownLatch是使用AQS实现的线程等待,
本文即通过实战和源码分析,探究CountDownLatch是如何利用AQS实现线程等待的,
帮助读者轻松应对知识交流与考核。

2 CountDownLatch

同步辅助工具,允许一个或多个线程等待其他线程操作完成。
CountDownLatch通过给定同步状态数量初始化,即state数量(参见AQS,AbstractQueuedSynchronizer)。
执行过程中,会通过调用countDown方法来阻塞await方法,直到同步状态state为0时,释放所有阻塞的线程,立即返回结果。
CountDownLatch的同步状态数量是一次性的操作,无法重置次数。
如果需要多次使用计数,可以使用CyclicBarrier。
CountDownLatch是通用的同步工具,用途广泛。
CountDownLatch初始化同步状态数量为1,可以作为一个简单的开/关锁存器或门:所有调用await方法的线程都在门处等待,直到线程调用countDown方法开启门。
CountDownLatch初始化同步状态数量为N,线程等待N个线程完成操作或某个线程完成N次。
CountDownLatch非常有用的一个特性是:执行前不要求调用countDown的线程等待同步状态数量达到0,仅仅是通过await方法阻止线程通过,直到所有任务都完成,一次性通过所有线程。
内存一致性影响:同步状态数量达到0之前,先调用countDown()的线程执行优先于其他线程调用await获取结果。

2.1 测试样例

CountDownLatch两个核心方法:countDown和await,
其中,
countDown同步状态数量减1,CountDownLatch继承AbstractQueuedSynchronizer(AQS),通过state属性检测是否获取或释放当前资源,所以,countDown方法主要的任务就是减少state值,释放资源。
await则是阻塞线程,state未达到0时持续自旋,等待,最终所有线程任务执行完成后立即返回,继续执行后面的逻辑。
下面给出具体的测试样例,新建CountDownLatch,初始化同步状态数state=3,执行三次countDown,释放线程,继续执行。

package com.monkey.java_study.juc;

import org.apache.commons.lang3.time.StopWatch;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.Random;
import java.util.concurrent.CountDownLatch;

/**
 * CountDownLatch测试.
 *
 * @author xindaqi
 * @since 2023-02-20 15:00
 */
public class CountDownLatchTest {

    private static final Logger logger = LoggerFactory.getLogger(CountDownLatchTest.class);

    /**
     * 线程执行器,
     * 执行countDown,计数器减一
     */
    static class ThreadRunner implements Runnable {

        private final CountDownLatch countDownLatch;

        public ThreadRunner(CountDownLatch countDownLatch) {
            this.countDownLatch = countDownLatch;
        }

        @Override
        public void run() {
            try {
                StopWatch stopWatch = new StopWatch();
                stopWatch.start();
                Random random = new Random();
                int randomBound = 1000;
                Thread.sleep(random.nextInt(randomBound));
                stopWatch.stop();
                logger.info(">>>>>>>>{}, time cost:{}", Thread.currentThread().getName(), stopWatch.formatTime());
                countDownLatch.countDown();
            } catch (Exception ex) {
                throw new RuntimeException(ex);
            }

        }
    }

    public static void main(String[] args) {
        StopWatch stopWatch = new StopWatch();
        stopWatch.start();
        int threadCount = 3;
        CountDownLatch countDownLatch = new CountDownLatch(threadCount);
        for (int i = 0; i < 3; i++) {
            new Thread(new ThreadRunner(countDownLatch)).start();
        }
        try {
            countDownLatch.await();
            stopWatch.stop();
            logger.info(">>>>>>>>Time cost:{}", stopWatch.formatTime());
        } catch (Exception ex) {
            throw new RuntimeException(ex);
        }
    }
}

测试结果如下图所示,
由结果可知,三个线程各自执行,通过不同的延迟时间模拟程序执行的时间,
执行耗时最长的Thread-2执行结束后,释放线程,继续执行后续的逻辑。
在这里插入图片描述

3 源码分析

下面根据源码分析CountDownLatch是如何实现线程等待以及释放资源的。

3.1 初始化

首先从初始化CountDownLatch开始讲解,源码如下图所示,
由源码可知,通过count参数初始化AQS的同步状态数state,
Sync即CountDownLatch中继承AbstractQueuedSynchronizer的内部类,
通过setState(count)初始化state,这个放在Sync中讲,先提示一下。
位置:java.util.concurrent.CountDownLatch#CountDownLatch
在这里插入图片描述

3.2 Sync

下面进入Sync源码,如下图所示,
由图可知,Sync是CountDownLatch的内部私有静态类,继承AbstractQueuedSynchronizer类,
构造函数参数为count,用于初始化state,即setState(count),
通过getCount()调用getState方法获取同步状态数量。
同时重写了tryAcquireShared和tryReleaseShared两个方法,
用于获取共享资源和释放共享资源。
位置:java.util.concurrent.CountDownLatch.Sync
在这里插入图片描述
获取共享资源和释放共享资源的源码如下图所示,
由图可知,tryAcquireShared通过获取同步状态数量标识当前资源状态,
同步状态不为0时,返回-1,为0时返回1,为后续提供操作标识,后文会反复用到,到时详解。
在这里插入图片描述

3.3 countDown

同步状态减1通过countDown方法实现,源码如下图所示,
由图可知,该方法调用了releaseShared方法,实现状态减1,这里还看不出具体的逻辑,
接着往下看,进入方法releaseShared。
位置:java.util.concurrent.CountDownLatch#countDown
在这里插入图片描述

3.3.1 releaseShared

releaseShared源码如下图所示,由图可知,
判断条件中调用了tryReleaseShared方法,为true时才会调用doReleaseShared,否则,直接返回。
其中,tryReleasShared方法由前文可知,
Sync内部类继承AQS,重写了该方法,实现同步状态state减1,state=0返回fasle,否则返回true,
由此可知,releaseShared方法通过tryReleasShared实现状态减1,在state不为0时,不会调用doReleaseShared方法。
当state=0时,才会调用doReleaseShared方法,此时,所有线程已经完成任务,将要全部通过latch(门闩),
由上述可推测,doReleaseShared用于释放资源。
这个方法并没有实现自旋等待,而是在await方法中进行等待的,在state没有达到0时,不会让线程通过。
位置:java.util.concurrent.locks.AbstractQueuedSynchronizer#releaseShared
在这里插入图片描述
tryReleaseShared源码标注如下图所示,
由图可知,通过自旋的方式使同步状态减1,state=0返回true,否则返回false,供后续方法判断。
位置:java.util.concurrent.CountDownLatch.Sync#tryReleaseShared
在这里插入图片描述

doReleaseShared则是释放资源,源码如下图所示,
由图可知,通过修改节点等待状态(waitStatus)为0,以及取消标记unparkSuccessor释放线程(这个放在后面讲),
通过自旋的方式,依次释放节点标记的线程资源,最终释放所有标记的资源。
位置:java.util.concurrent.locks.AbstractQueuedSynchronizer#doReleaseShared
在这里插入图片描述
unparkSuccessor释放节点标记的线程,源码如下图所示,
由源码可知,获取当前节点的后继节点,
后继节点不为空时,通过LockSupport.unpark释放线程标记,
当节点的后继节点不为空时,会释放当前节点标记的线程,即节点不再等待线程,直接释放,
实现最后一个线程执行结束后,取消节点对线程的占用,释放资源。
这里直接取消线程标记,也是导致同步状态state无法重用的直接原因,已经取消标记,后续自然无法使用,
同时,state置为0后,没有重新配置,所以依旧不能重用state。
在这里插入图片描述

3.4 await

综上分析,CountDownLatch控制线程等待的逻辑是在await中实现的,
因为,上面的逻辑只是实现了同步状态减1,以及释放线程标记等工作,没有阻塞线程,
所以,在await阻塞线程,当所有线程均完成各自逻辑时,即同步状态state=0,才开启“闸门”,释放锁。
await源码如下图所示,由图可知,await调用acquireSharedInterruptly方法,
当同步状态state=0时,立即返回。
位置:java.util.concurrent.CountDownLatch#await()
在这里插入图片描述

3.4.1 acquireSharedInterruptibly

线程等待。同步状态state不为0时,线程间相互等待。
acquireSharedInterruptibly源码如下图所示,
由图可知,判断条件通过tryAcquireShared获取,当state不为0时,返回-1,参见上文的tryAcquireShared方法,
调用doAcquireSharedInterrutly方法,实现线程等待。
位置:java.util.concurrent.locks.AbstractQueuedSynchronizer#acquireSharedInterruptibly
在这里插入图片描述

3.4.2 doAcquireSharedInterruptily

线程等待具体实现,源码如下图所示,
由图可知,每次调用都会向队列插入一个新的节点Node.SHARED,
然后自旋,即线程等待,当前节点的前驱节点为头节点时,获取同步状态state,
state不为0时,tryAcquireShared=-1,不会跳出自旋,
继续等待countDown操作,
当countDown使state减为0时,tryAcquireShared返回1,开始执行r>=0后续的逻辑,
跳出自旋,立即返回,即await立即返回,继续执行await后续的逻辑,释放线程等待。

位置:java.util.concurrent.locks.AbstractQueuedSynchronizer#doAcquireSharedInterruptibly

在这里插入图片描述
上面提到获取当前节点的前驱节点,通过predecessor方法,
源码如下图所示,由图可知,返回节点的前驱节点。
位置:java.util.concurrent.locks.AbstractQueuedSynchronizer.Node#predecessor
在这里插入图片描述

至此完成CountDownLatch多线程同步状态递减以及相互等待的分析。

4 小结

(1)CountDownLatch是一次性的线程等待工具,实现多线程相互等待,直到所有线程均完成执行逻辑,用完即抛,无法重用;
(2)countDown方法使同步状态减1,当一个线程执行完成后,调用countDown,标识当前线程已完成,当state=0时,释放节点的线程占用,这也是CountDownLatch无法重用的直接原因,当然,没有重复保存state也是另一个原因;
(3)await是线程等待的核心,当同步状态state不为0时,会自旋等待,而不触发跳出自旋,当state=0时,才会立即返回,跳出自旋,结束线程等待;
(4)当某个线程出现异常时,这里需要格外注意,为保证程序正常中断,需要手动捕获异常,同时调用countDown方法,或者使用重试机制,否则线程不会退出,因为state最终的状态不为0,有一个线程线程而没有减1。

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

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

相关文章

QML 元素布局

定位器&#xff1a;是QtQuick模块中的提供的&#xff0c;有以下三种 Row 行定位器Column 列定位器Grid 网格定位器Flow 流动定位器常用属性: spacing间距 Row &#xff08;行定位器&#xff09; 按照行的方排列 //行定位器 Row{spacing: 5//设置间距Rectangle{width: 100he…

[蓝桥杯] 二分与前缀和习题练习

文章目录 一、二分查找习题练习 1、1 数的范围 1、1、1 题目描述 1、1、2 题解关键思路与解答 1、2 机器人跳跃问题 1、2、1 题目描述 1、2、2 题解关键思路与解答 1、3 四平方和 1、3、1 题目描述 1、3、2 题解关键思路与解答 二、前缀和习题练习 2、1 前缀和 2、1、1 题目描述…

《操作系统》——第二章 进程与线程

目录 2.1.1进程的概念、组成、特征 2.1.2进程的状态与转换、进程的组织 2.1.3进程控制 2.1.4进程通信 2.1.5线程的概念 2.1.6线程的实现方式和多线程模型 2.2.1调度的概念、层次 2.2.2进程调度的时机、切换与过程、方式 2.2.4调度算法的评价指标 2.2.5调度算法(1) 2…

1249 亲戚(并查集)

1249. 亲戚 题目 提交记录 讨论 题解 视频讲解或许你并不知道&#xff0c;你的某个朋友是你的亲戚。 他可能是你的曾祖父的外公的女婿的外甥女的表姐的孙子。 如果能得到完整的家谱&#xff0c;判断两个人是否是亲戚应该是可行的&#xff0c;但如果两个人的最近公共祖…

数据库之高级查询

注意&#xff1a;第一个包含空&#xff0c;第二句不包含空注意&#xff1a;第二句是错的&#xff0c;聚合函数不能出现在where中。注意&#xff1a;相当于&#xff0c;按照分组属性&#xff0c;求出每个组的聚合函数值&#xff0c;所以肯定不能放单个属性有冲突with rollup是最…

MyBatis - 05 - 封装SqlSessionUtil工具类(用于获取SqlSession对象)并测试功能

文章目录1.新建SqlSessionUtils工具类2.编写静态方法3.项目结构及代码项目结构数据库和表pom.xmlParameterMapper接口&#xff1a;User类&#xff1a;ParameterMapper.xmljdbc.propertieslog4j.xml:mybatis-config.xml:ParameterMapperTest测试类&#xff1a;测试结果1.新建Sql…

leetcode打卡-回溯I

77. 组合 leetcode题目链接&#xff1a;https://leetcode.cn/problems/combinations/ leetcode AC记录&#xff1a; 代码如下&#xff1a; public List<List<Integer>> combine(int n, int k) {List<List<Integer>> res new ArrayList<>(16…

操作系统期末复习

操作系统概论 文章目录操作系统概论操作系统的目标&#xff1a;基本特征:主要功能发展操作系统的运行机制时钟管理中断机制&#xff1a;指令程序处理机状态原语&#xff1a;由若干指令组成的程序段&#xff0c;完成特定功能系统数据结构系统调用体系结构进程--资源分配和调度的…

[黑马程序员SSM框架教程] Spring-22 注解开发依赖注入

1.自动装配是基于暴力反射对私有属性进行装配的&#xff0c;所以不需要setter方法。打破了IOC提供对象&#xff0c;我提供入口的思想。现在不用提供入口也能实现。 2. Qualifier必须依赖注解Autowired&#xff0c;当自动装配的接口类型有多个实现类时使用&#xff0c; 3. Autow…

Linux基础命令-du查看文件的大小

文章目录 du 命令介绍 语法格式 基本参数 参考实例 1&#xff09;以人类可读形式显示指定的文件大小 2&#xff09;显示当前目录下所有文件大小 3&#xff09;只显示目录的大小 4&#xff09;显示根下哪个目录文件最大 5&#xff09;显示所有文件的大小 6&#xff0…

layui框架学习(11:徽章)

应用程序有新增内容、未读消息时&#xff0c;会在按钮或菜单中添加红点或带数字的点状或方状图形&#xff0c;用户看到就知道有新内容&#xff0c;如下图所示QQ邮箱的截图&#xff0c;会通过红色圆点或带NEW的方框提醒用户有新内容可以查看。   CSDN用户如果有新消息&#x…

产品经理有必要考个 PMP吗?(含PMP资料)

现在基本上做产品的都有一个PMP证件&#xff0c;从结果导向来说&#xff0c;不对口不会有这么大范围的人来考&#xff0c;但是需要因地制宜&#xff0c;在公司内部里&#xff0c;标准程序并不流畅&#xff0c;产品和项目并不规范&#xff0c;关系错综复杂。 而产品经理的职能又…

【Java学习】初识Java

JavaSEJava初识1. Java简介2.Java环境的安装与配置3. 开发第一个Java程序Java初识 学前疑问&#xff1a;&#xff08;带着疑问去学习&#xff0c;在学习中自行探索答案&#xff09; Java是什么&#xff1f;能做什么&#xff1f;发展前景如何&#xff1f;需要学习哪些内容&…

腾讯一面—Android 系统启动流程详解

正文AMS 是 Android 中最核心的服务之一&#xff0c;主要负责系统中四大组件的启动、切换、调度及应用进程的管理和调度等工作&#xff0c;其职责与操作系统中的进程管理和调度模块相类似&#xff0c;它本身也是一个 Binder 的实现类&#xff0c;应用进程能通过 Binder 机制调用…

ESPRESSIF ESP32 开发环境搭建

1.下载Espressif-IDE并安装&#xff0c;安装的时候直接下一步就行&#xff0c;下载地址为 IDE下载地址 2.新建项目 1&#xff0c;首先磁盘新建文件夹&#xff0c;用来保存项目 2&#xff0c;File ->New ->乐鑫IDF项目-> 工程名字test->保存目录就是自己建立的哪个…

EasyRecovery16MAC苹果版本Photo最新版数据恢复软件

无论是在工作学习中&#xff0c;还是在生活中&#xff0c;Word、Excle等办公软件都是大家很常用的。我们在使用电脑的过程中&#xff0c;有时会因自己的误删或电脑故障&#xff0c;从而导致我们所写的文档丢失了。出现这样的大家不要着急&#xff0c;今天小编就给大家推荐一款可…

nacos 单机集群搭建及常用生产环境配置 | Spring Cloud 3

一、Nacos 概览 Nacos /nɑ:kəʊs/ 是 Dynamic Naming and Configuration Service的首字母简称&#xff0c;一个更易于构建云原生应用的动态服务发现、配置管理和服务管理平台。 Nacos 致力于帮助您发现、配置和管理微服务。Nacos 提供了一组简单易用的特性集&#xff0c;帮…

《程序员思维修炼》速读笔记

文章目录书籍信息概览绪论从新手到专家的历程认识大脑利用右脑调试大脑主动学习积累经验控制注意力超越专家图解书籍信息 书名&#xff1a;《程序员思维修炼&#xff08;修订版&#xff09;》 作者&#xff1a;[美] Andy Hunt 概览 绪论 再提“实用”关注情境所有人都关注这…

kafka使用入门案例与踩坑记录

每次用到kafka时都会出现各种奇怪的问题&#xff0c;综合实践&#xff0c;下面汇总下主要操作步骤&#xff1a; Docker镜像形式启动 zookeeper启动 docker run -d --name zookeeper -p 2181:2181 -t wurstmeister/zookeeperkafka启动 docker run --name kafka01 -p 9092:909…

Apk转Aab(Android-App-Bundle)

这篇文章是参考Apk转Aab(Android-App-Bundle)_YoungBillsohu的博客-CSDN博客 基本照着这个大佬的步骤来就行&#xff0c;但是要注意的是apkTool最好是下新的&#xff0c;否则&#xff0c;会出现说一堆无语的错误&#xff0c;然后导致AAPT2关联资源的时候报错 类似这样的&#…