Java中的CountDownLatch同步工具类使用解析

 更新时间:2023年12月16日 09:53:14   作者:niulx111  
这篇文章主要介绍了Java中的CountDownLatch使用解析,CountDownLatch初始化的时候必须指定一个count,await方法会一直阻塞直到调用countdown方法,count为0,当count为0时,所有的等待线程都会被释放,需要的朋友可以参考下

CountDownLatch同步工具类

CountDownLatch初始化的时候必须指定一个count,await方法会一直阻塞直到调用countdown方法,count为0,当count为0时,所有的等待线程都会被释放。

count是不能被重置的,如果想重复使用count,可以考虑CyclicBarrier。

CountDownLatch是一个同步工具类,用来协调多个线程之间的同步,或者是线程之间的通信。

CountDownLatch可以使主线程等待子线程完成自己的任务之后在继续执行,count为线程的数量,每当线程完成一个任务后,count减一,当count为0时,表示所有的任务都已经完成,这时主线程就可以继续执行。

CountDownLatch的构造函数

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

初始化的时候必须指定一个大于等于0的count,否则会抛出异常。

        Sync(int count) {
            setState(count);
        }

CountDownLatch的同步是使用了AQS的状态代表count。

看一下它的countdown方法。

    /**
     * 如果count为0时,释放所有的等待线程
     * 如果当前的count比0大就要递减
     * 如果当前count为0什么也不做
     */
    public void countDown() {
        sync.releaseShared(1);
    }

本质还是调用了AQS的releaseShared方法。

    public final boolean releaseShared(int arg) {
        if (tryReleaseShared(arg)) {
            doReleaseShared();
            return true;
        }
        return false;
    }
protected boolean tryReleaseShared(int releases) {
            // Decrement count; signal when transition to zero
            for (;;) {
                int c = getState();
                if (c == 0)
                    return false;
                int nextc = c-1;
                if (compareAndSetState(c, nextc))
                    return nextc == 0;
            }
        }

countdown就是释放锁的操作,每被调用一次,state就减一。首先尝试释放锁,利用CAS设置state,如果state为0,说明所有的子线程都完成了操作,这是就要唤醒在同步队列上的其他线程。

再来看一下await方法。

    public void await() throws InterruptedException {
        sync.acquireSharedInterruptibly(1);
    }
    public final void acquireSharedInterruptibly(int arg)
            throws InterruptedException {
        if (Thread.interrupted())
            throw new InterruptedException();
        if (tryAcquireShared(arg) < 0)
            doAcquireSharedInterruptibly(arg);
    }
        protected int tryAcquireShared(int acquires) {
            return (getState() == 0) ? 1 : -1;
        }

尝试获取锁,如果state为0,表示获取锁成功,如果state不为0,表示获取锁失败,调用doAcquireSharedInterruptibly方法阻塞直到成功获取到锁。

CountDownLatch的使用

public static void main(String[] args) throws Exception {
		java.util.concurrent.CountDownLatch c = new java.util.concurrent.CountDownLatch(3);
		ExecutorService es = Executors.newCachedThreadPool();
		for (int i = 0; i < 3; i++) {
			es.execute(new Task(i, c));
		}
		c.await();
		System.out.println("主线程执行任务");
	}
	public static class Task implements Runnable {
		private int m;
		private java.util.concurrent.CountDownLatch c;
		public Task(int m,java.util.concurrent.CountDownLatch c) {
			this.m = m;
			this.c = c;
		}
		@Override
		public void run() {
			System.out.println("子线程"+m+"完成了任务");
			c.countDown();
		}
	}

执行结果

子线程1完成了任务
子线程0完成了任务
子线程2完成了任务
主线程执行任务

在多线程中,子线程需要完成各自的任务后,主线程才能利用子线程的结果进行整合,我们可以考虑CountDownLatch来控制并发。

CountDownLatch只是一个同步辅助类,当CountDownLatch的计数器未到0之前,所有调用await的方法都会阻塞,只有计数器为0,线程才能继续往下执行。CountDownLatch的计数器是不可重用的。

到此这篇关于Java中的CountDownLatch同步工具类使用解析的文章就介绍到这了,更多相关CountDownLatch同步工具类解析内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 一文秒懂springboot druid 配置

    一文秒懂springboot druid 配置

    Druid是阿里巴巴开发的一个连接池,他提供了一个高效、功能强大、可扩展性好的数据库连接池,区别于hikari,今天通过本文给大家分享springboot druid 配置教程,需要的朋友参考下吧
    2021-08-08
  • Java动态脚本Groovy

    Java动态脚本Groovy

    本文介绍了Java动态脚本Groovy,Groovy是用于Java虚拟机的一种敏捷的动态语言,它是一种成熟的面向对象编程语言,既可以用于面向对象编程,又可以用作纯粹的脚本语言。使用该种语言不必编写过多的代码,同时又具有闭包和动态语言中的其他特性,需要的朋友可以参考一下
    2021-12-12
  • 详解Java分布式系统中session一致性问题

    详解Java分布式系统中session一致性问题

    这篇文章主要介绍了Java分布式系统中session一致性问题,对分布式系统感兴趣的同学,要仔细看一下
    2021-04-04
  • Springboot实现导入导出Excel的方法

    Springboot实现导入导出Excel的方法

    今天带各位小伙伴学习Springboot实现导入导出Excel的方法,文中有非常详细的介绍,对正在学习java的小伙伴们有很好地帮助,需要的朋友可以参考下
    2021-05-05
  • java编程abstract类和方法详解

    java编程abstract类和方法详解

    这篇文章主要介绍了java编程abstract类和方法详解,具有一定借鉴价值,需要的朋友可以参考下。
    2017-12-12
  • 深入了解Spring中最常用的11个扩展点

    深入了解Spring中最常用的11个扩展点

    我们一说到spring,可能第一个想到的是 IOC(控制反转) 和 AOP(面向切面编程)。除此之外,我们在使用spring的过程中,有没有发现它的扩展能力非常强。今天就来跟大家一起聊聊,在Spring中最常用的11个扩展点
    2022-09-09
  • Java编写程序之输入一个数字实现该数字阶乘的计算

    Java编写程序之输入一个数字实现该数字阶乘的计算

    这篇文章主要介绍了Java编写程序之输入一个数字实现该数字阶乘的计算,本文通过实例代码给大家介绍的非常想详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-02-02
  • SpringBoot项目实现日志打印SQL的常用方法(包括SQL语句和参数)

    SpringBoot项目实现日志打印SQL的常用方法(包括SQL语句和参数)

    有时候遇到问题需要根据我们编写的SQL进行分析,但如果不进行一些开发或者配置的话,这些SQL是不会打印到控制台的,它们默认是隐藏的。下面给大家介绍几种常用的方法,感兴趣的朋友跟随小编一起看看吧
    2024-04-04
  • Java中类加载过程全面解析

    Java中类加载过程全面解析

    这篇文章主要介绍了Java中类加载过程全面解析,具有一定参考价值,需要的朋友可以了解下。
    2017-11-11
  • java正则表达式对特殊字符的转义以及异常现象

    java正则表达式对特殊字符的转义以及异常现象

    这篇文章主要给大家介绍了关于java正则表达式对特殊字符的转义以及异常现象的相关资料,在Java中使用正则表达式匹配特殊字符也需要使用转义字符"\",需要的朋友可以参考下
    2023-06-06

最新评论