Java并发编程之CountDownLatch闭锁

简介: Java并发编程之CountDownLatch闭锁

CountDownLatch



  1. 典型应用场景:主线程启动多个子线程同时执行业务逻辑,所有子线程都执行完毕,再唤醒主线程继续执行。


  1. 例子:


public class CountDownLatchTest {
    /**
     * 计数器,初始为0
     */
    private Integer count = 0;
    public Integer getCount(){
        return count;
    }
    /**
     * 执行+1操作
     */
    public void add(){
        count++;
    }
    public static void main(String[] args){
        CountDownLatchTest test = new CountDownLatchTest();
        // 线程个数
        int threadCount = 3;
        //初始化工作线程的个数,并用CountDownLatch管理
        CountDownLatch countDownLatch = new CountDownLatch(threadCount);
        for(int i=0;i<threadCount;i++) {
            new Thread(() -> {
                test.add();
                countDownLatch.countDown();
            }).start();
        }
        try {
            //等待所有线程执行完毕,在所有线程都执行完毕之前主线程会阻塞
            countDownLatch.await();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(test.getCount());
    }
}


主线程启动了3个子线程执行add操作,等待3个子线程都执行完毕了,主线程继续执行打印最终的执行结果为:3。


3.具体实现原理:


public class CountDownLatch {
 //继承于AQS的同步器
 private static final class Sync extends AbstractQueuedSynchronizer {
        private static final long serialVersionUID = 4982264981922014374L;
        //有参构造函数,count记录了共享资源的个数
        Sync(int count) {
            setState(count);
        }
        //获取当前共享资源的个数
        int getCount() {
            return getState();
        }
        /**
        * 尝试以共享方式获取资源
        * @return 1表示获取成功,-1表示获取失败
        */
        protected int tryAcquireShared(int acquires) {
            //如果当前资源个数为0,则表示获取成功,否则表示失败
            return (getState() == 0) ? 1 : -1;
        }
       /**
        * 尝试以共享方式释放资源
        * @return true表示释放成功,false表示释放失败
        */
        protected boolean tryReleaseShared(int releases) {
            // 对当前资源执行-1操作
            for (;;) {
                int c = getState();
                if (c == 0)
                    return false;
                int nextc = c-1;
                //CAS更新资源个数,CAS失败表示有其他线程竞争,此时需要重试
                if (compareAndSetState(c, nextc))
                    //执行-1操作后,如果资源个数为0,则表示释放成功
                    return nextc == 0;
            }
        }
    }
    private final Sync sync;
    //有参构造函数,可以看到CountDownLatch中禁用了默认构造函数,意味着必须传入资源个数
    public CountDownLatch(int count) {
        if (count < 0) throw new IllegalArgumentException("count < 0");
        this.sync = new Sync(count);
    }
    //等待操作,此方法会使调用线程阻塞,直到其他调用countdown的方法都执行完毕
    public void await() throws InterruptedException {
        //此处调用的是AQS的acquireSharedInterruptibly方法,下文会具体分析
        sync.acquireSharedInterruptibly(1);
    }
    //和await()类似,但是有一个等待的超时时间,过了超时时间会自动取消等待
    public boolean await(long timeout, TimeUnit unit)
        throws InterruptedException {
        return sync.tryAcquireSharedNanos(1, unit.toNanos(timeout));
    }
    //将state的值-1,当getState()==0时,会唤醒调用await()线程
    public void countDown() {
        //调用AQS的releaseShared方法,下文会具体分析
        sync.releaseShared(1);
    }
    //获取当前资源的个数
    public long getCount() {
        return sync.getCount();
    }
}


4.CountDownLatch中用到的AQS的核心方法:


public final void acquireSharedInterruptibly(int arg)
    throws InterruptedException {
    if (Thread.interrupted())
        throw new InterruptedException();
    //尝试获取资源失败(tryAcquireShared的返回值<0),会将当前线程阻塞并排队等待
    if (tryAcquireShared(arg) < 0)
        //该方法会将当前线程阻塞,并放入AQS的同步队列等待,此处不再分析
        doAcquireSharedInterruptibly(arg);
}
//释放共享资源
public final boolean releaseShared(int arg) {
    //尝试释放共享资源成功时(此处要结合CountDownLatch提供的tryReleaseShared方法理解),进行具体的释放操作
    if (tryReleaseShared(arg)) {
        //AQS提供的执行具体的资源释放操作,会唤醒调用await()方法的线程
        doReleaseShared();
        return true;
    }
    return false;
}


5.总结CountDownLatch使用AQS的state变量作为状态计数器,执行countdown操作的线程会将计数器减1,当前计数器的值为0时(getState()==0),会唤醒执行await操作的线程继续执行。

目录
相关文章
|
5月前
|
IDE Java 编译器
java编程最基础学习
Java入门需掌握:环境搭建、基础语法、面向对象、数组集合与异常处理。通过实践编写简单程序,逐步深入学习,打牢编程基础。
321 1
|
5月前
|
Java
如何在Java中进行多线程编程
Java多线程编程常用方式包括:继承Thread类、实现Runnable接口、Callable接口(可返回结果)及使用线程池。推荐线程池以提升性能,避免频繁创建线程。结合同步与通信机制,可有效管理并发任务。
244 6
|
5月前
|
安全 前端开发 Java
从反射到方法句柄:深入探索Java动态编程的终极解决方案
从反射到方法句柄,Java 动态编程不断演进。方法句柄以强类型、低开销、易优化的特性,解决反射性能差、类型弱、安全性低等问题,结合 `invokedynamic` 成为支撑 Lambda 与动态语言的终极方案。
241 0
|
6月前
|
SQL Java 数据库
2025 年 Java 从零基础小白到编程高手的详细学习路线攻略
2025年Java学习路线涵盖基础语法、面向对象、数据库、JavaWeb、Spring全家桶、分布式、云原生与高并发技术,结合实战项目与源码分析,助力零基础学员系统掌握Java开发技能,从入门到精通,全面提升竞争力,顺利进阶编程高手。
1104 2
|
6月前
|
Java 开发者
Java并发编程:CountDownLatch实战解析
Java并发编程:CountDownLatch实战解析
543 100
|
6月前
|
NoSQL Java 关系型数据库
超全 Java 学习路线,帮你系统掌握编程的超详细 Java 学习路线
本文为超全Java学习路线,涵盖基础语法、面向对象编程、数据结构与算法、多线程、JVM原理、主流框架(如Spring Boot)、数据库(MySQL、Redis)及项目实战等内容,助力从零基础到企业级开发高手的进阶之路。
470 1
|
6月前
|
算法 Java
Java多线程编程:实现线程间数据共享机制
以上就是Java中几种主要处理多线程序列化资源以及协调各自独立运行但需相互配合以完成任务threads 的技术手段与策略。正确应用上述技术将大大增强你程序稳定性与效率同时也降低bug出现率因此深刻理解每项技术背后理论至关重要.
449 16
|
5月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
276 1
|
5月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
295 1
|
6月前
|
数据采集 存储 弹性计算
高并发Java爬虫的瓶颈分析与动态线程优化方案
高并发Java爬虫的瓶颈分析与动态线程优化方案