开发基础
开发基础笔记
mybatis-plus通用的service方法
通过jenkins配置前后端自动打包及发布
mybatis-plus实现分页的方式
Java极客技术公众号PDF
狂神说SpringBoot
Hashids java 版使用
SpringBoot
1、Web快速开发
2、结果集的封装
3、集成MyBatis实现数据库操作
4、Springboot @Validated参数校验
5、SpringBoot全局异常处理
6、拦截器HandlerInterceptor
7、集成Swagger实现API自动生成
8、集成knife4j实现API自动生成
9、Springboot集成MyBatis-Plus快速入门
10、springboot自定义注解及AOP切面使用
11、使用Shiro实现登陆和权限认证,基于MyBatis
12、集成SpringSecurity实现授权认证
13、SpringBoot集成EasyExcel实现数据导入与导出
14、Spring Task定时任务的实现
15、Quartz快速上手与实践
16、如何用代码实现Spring IOC
17、SpringBoot集成JWT,实现接口的鉴权交互
SpringCloud
Nacos作为服务注册中心
seata1.6.1 结合springcloud实现分布锁的技术笔记
一些技术博客推荐
前端面试相关
看这一篇就够了
java.util包常用的类和接口
CountDownLatch介绍与使用
Common-lang3使用入门
Hutool简单使用入门
lombok 介绍及基本使用方法
git项目统计成员代码行数和提交的次数
mysql 逗号分隔的数据 like查询
使用sonar进行代码质量检查
线上使用jmeter进行压测的时候,使用Arthas诊断工具排查响应慢的接口
php结合phpstudy8、vscode开启xdebug进行代码调试
node-red使用入门
本文档使用 MrDoc 发布
-
+
首页
CountDownLatch介绍与使用
`CountDownLatch`是一个同步工具类,用来协调多个线程之间的同步,或者说起到线程之间通信的作用(非互斥)。 CountDownLatch 能够使一个线程在等待另外一些线程完成各自工作之后,再继续执行。使用一个计数器进行实现。计数器初始值为线程的数量。当每一个线程完成自己任务后,计数器的值就会减一。当计数器的值为0时,表示所有的线程都已经完成一些任务,然后在CountDownLatch上等待的线程就可以恢复执行接下来的任务。  ### CountDownLatch 的使用 CountDownLatch类使用起来非常简单。 Class 位于:`java.util.concurrent.CountDownLatch` 下面简单介绍它的构造方法和常用方法。 ### 构造方法 CountDownLatch只提供了一个构造方法: ```java // count 为初始计数值 public CountDownLatch(int count) { // …… } ``` ### 常用方法 ```java //常用方法1:调用await()方法的线程会被挂起,它会等待直到count值为0才继续执行 public void await() throws InterruptedException { // …… } // 常用方法2:和await()类似,只不过等待超时后count值还没变为0的话就会继续执行 public boolean await(long timeout, TimeUnit unit) throws InterruptedException { // …… } // 常用方法3:将count值减1 public void countDown() { // …… } ``` ### CountDownLatch 的应用场景 我们考虑一个场景:用户购买一个商品下单成功后,我们会给用户发送各种消息提示用户『购买成功』,比如发送邮件、微信消息、短信等。所有的消息都发送成功后,我们在后台记录一条消息表示成功。 当然我们可以使用单线程去完成,逐个完成每个操作,如下图所示:  但是这样效率就会非常低。如何解决单线程效率低的问题?当然是通过多线程啦。 使用多线程也会遇到一个问题,子线程消息还没发送完,主线程可能就已经打出『所有的消息都已经发送完毕啦』,这在逻辑上肯定是不对的。我们期望所有子线程发完消息主线程才会打印消息,怎么实现呢?CountDownLatch就可以解决这一类问题。  我们使用代码实现上面的需求。 ```java import java.util.concurrent.*; public class OrderServiceDemo { public static void main(String[] args) throws InterruptedException { System.out.println("main thread: Success to place an order"); int count = 3; CountDownLatch countDownLatch = new CountDownLatch(count); Executor executor = Executors.newFixedThreadPool(count); executor.execute(new MessageTask("email", countDownLatch)); executor.execute(new MessageTask("wechat", countDownLatch)); executor.execute(new MessageTask("sms", countDownLatch)); // 主线程阻塞,等待所有子线程发完消息 countDownLatch.await(); // 所有子线程已经发完消息,计数器为0,主线程恢复 System.out.println("main thread: all message has been sent"); } static class MessageTask implements Runnable { private String messageName; private CountDownLatch countDownLatch; public MessageTask(String messageName, CountDownLatch countDownLatch) { this.messageName = messageName; this.countDownLatch = countDownLatch; } @Override public void run() { try { // 线程发送消息 System.out.println("Send " + messageName); try { TimeUnit.SECONDS.sleep(1); } catch (InterruptedException e) { e.printStackTrace(); } } finally { // 发完消息计数器减 1 countDownLatch.countDown(); } } } } ``` 程序运行结果: ```text main thread: Success to place an order Send email Send wechat Send sms main thread: all message has been sent ``` 从运行结果可以看到主线程是在所有的子线程发送完消息后才打印,这符合我们的预期。 ### CountDownLatch 的限制 CountDownLatch是一次性的,计算器的值只能在构造方法中初始化一次,之后没有任何机制再次对其设置值,当CountDownLatch使用完毕后,它不能再次被使用。
superadmin
2024年11月21日 15:07
转发文档
收藏文档
上一篇
下一篇
手机扫码
复制链接
手机扫一扫转发分享
复制链接
Markdown文件
分享
链接
类型
密码
更新密码