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操作的线程继续执行。

目录
相关文章
|
4天前
|
安全 Java UED
Java中的多线程编程:从基础到实践
本文深入探讨了Java中的多线程编程,包括线程的创建、生命周期管理以及同步机制。通过实例展示了如何使用Thread类和Runnable接口来创建线程,讨论了线程安全问题及解决策略,如使用synchronized关键字和ReentrantLock类。文章还涵盖了线程间通信的方式,包括wait()、notify()和notifyAll()方法,以及如何避免死锁。此外,还介绍了高级并发工具如CountDownLatch和CyclicBarrier的使用方法。通过综合运用这些技术,可以有效提高多线程程序的性能和可靠性。
|
4天前
|
缓存 Java UED
Java中的多线程编程:从基础到实践
【10月更文挑战第13天】 Java作为一门跨平台的编程语言,其强大的多线程能力一直是其核心优势之一。本文将从最基础的概念讲起,逐步深入探讨Java多线程的实现方式及其应用场景,通过实例讲解帮助读者更好地理解和应用这一技术。
19 3
|
4天前
|
Java 开发者
在Java编程中,正确的命名规范不仅能提升代码的可读性和可维护性,还能有效避免命名冲突。
【10月更文挑战第13天】在Java编程中,正确的命名规范不仅能提升代码的可读性和可维护性,还能有效避免命名冲突。本文将带你深入了解Java命名规则,包括标识符的基本规则、变量和方法的命名方式、常量的命名习惯以及如何避免关键字冲突,通过实例解析,助你写出更规范、优雅的代码。
25 3
|
4天前
|
Java 程序员
在Java编程中,关键字不仅是简单的词汇,更是赋予代码强大功能的“魔法咒语”。
【10月更文挑战第13天】在Java编程中,关键字不仅是简单的词汇,更是赋予代码强大功能的“魔法咒语”。本文介绍了Java关键字的基本概念及其重要性,并通过定义类和对象、控制流程、访问修饰符等示例,展示了关键字的实际应用。掌握这些关键字,是成为优秀Java程序员的基础。
11 3
|
4天前
|
Java 程序员 编译器
在Java编程中,保留字(如class、int、for等)是具有特定语法意义的预定义词汇,被语言本身占用,不能用作变量名、方法名或类名。
在Java编程中,保留字(如class、int、for等)是具有特定语法意义的预定义词汇,被语言本身占用,不能用作变量名、方法名或类名。本文通过示例详细解析了保留字的定义、作用及与自定义标识符的区别,帮助开发者避免因误用保留字而导致的编译错误,确保代码的正确性和可读性。
15 3
|
4天前
|
算法 Java
在Java编程中,关键字和保留字是基础且重要的组成部分,正确理解和使用它们
【10月更文挑战第13天】在Java编程中,关键字和保留字是基础且重要的组成部分。正确理解和使用它们,如class、int、for、while等,不仅能够避免语法错误,还能提升代码的可读性和执行效率。本指南将通过解答常见问题,帮助你掌握Java关键字的正确使用方法,以及如何避免误用保留字,使你的代码更加高效流畅。
19 3
|
3天前
|
存储 安全 Java
了解final关键字在Java并发编程领域的作用吗?
在Java并发编程中,`final`关键字不仅用于修饰变量、方法和类,还在多线程环境中确保对象状态的可见性和不变性。本文深入探讨了`final`关键字的作用,特别是其在final域重排序规则中的应用,以及如何防止对象的“部分创建”问题,确保线程安全。通过具体示例,文章详细解析了final域的写入和读取操作的重排序规则,以及这些规则在不同处理器上的实现差异。
了解final关键字在Java并发编程领域的作用吗?
|
4月前
|
Java C++
关于《Java并发编程之线程池十八问》的补充内容
【6月更文挑战第6天】关于《Java并发编程之线程池十八问》的补充内容
48 5
|
1月前
|
缓存 监控 Java
Java中的并发编程:理解并应用线程池
在Java的并发编程中,线程池是提高应用程序性能的关键工具。本文将深入探讨如何有效利用线程池来管理资源、提升效率和简化代码结构。我们将从基础概念出发,逐步介绍线程池的配置、使用场景以及最佳实践,帮助开发者更好地掌握并发编程的核心技巧。
|
3月前
|
安全 Java 开发者
Java中的并发编程:深入理解线程池
在Java的并发编程中,线程池是管理资源和任务执行的核心。本文将揭示线程池的内部机制,探讨如何高效利用这一工具来优化程序的性能与响应速度。通过具体案例分析,我们将学习如何根据不同的应用场景选择合适的线程池类型及其参数配置,以及如何避免常见的并发陷阱。
52 1