《重学Java高并发》同步转异步编程技巧与实战运用

简介: 《重学Java高并发》同步转异步编程技巧与实战运用

1、线程池+Future模式


笔者在公司中负责开发某一个产品时,需要实现一个告警模块,告警通知方式需要为钉钉群、电话短信等方式,并且及时时单一的告警方式,例如钉钉群告警,也需要同时发送到多个群(监控中心、业务项目组钉钉群),使监控告警能真正通知到各个相关方,确保人工及时处理跟进,避免事态进一步发展。


发送钉钉群告警信息的时序图如下:

8cde7b9f96dbd008e88b05cbee1595f7.png

发送到不同的钉钉群,这个过程完成可以并发,并发同步等待发送结果,即这个过程是一个同步场景,但可以转化成异步(多并发),即经典的同步转异步


要实现这个功能,我想大家第一时间会想到使用线程池+Future模式。

0502d7feda738c10f599cc7b7c5f618f.png

代码实现的关键点如下:


  • 创建一个线程池,请大家一定要使用自定义线程工厂,为创建的线程命名,如代码@2.
  • 两个任务,提交到线程池中,此过程是一个异步。注意:SendDingDingTalk是实现java.util.concurrent.Callable,即带返回值的任务。
  • 提交到线程池中的任务如果实现了java.util.concurrent.Callable,提交到线程池返回一个Future对象(凭证),通过调用Feture对象的get()方法,如果任务未完成,则会阻塞,即实现同步转异步调用,再转同步的调用效果,从而提高并发,提高性能。


2、CountDownLatch的妙用


使用线程池+Future模式,有时候会显得比较笨重,因为需要额外创建一个线程池,如果在一些轻量级场景(单线程+批量)场景下也希望将同步转异步,我们有其他办法没?


使用CountDownLatch!!!


接下来我们结合场景来说说如何使用CountDownLatch。


例如我们在对数据库数据进行清理时,通常会将数据进行分页(任务分批),然后创建多个子线程,主线程将任务分批后提交到子线程,等待子线程全部执行完成后,主线程打印执行日期,其时序图如下所示:

a303d5131b6c987e976ace315eee2e3d.png

主线程如何得知子线程执行完毕呢?在java中,通常的方案是 join,但juc框架中的CountDownLatch实现了与Thread.join相同的语义,其伪代码实现如下:

97df1a14c20d3eae7c45a81158770fd7.png

使用CountDownLatch的要点如下:


  • 首先会创建一个CountDownLatch,并且指定计数器,通常为子线程的个数。
  • 然后主线程向子线程提交任务,并且子线程在完成自己的工作后,调用CountDownLatch的countDown,计数器降一,“以此来表示子线程已处理完毕”。
  • 各个子线程异步执行,但最终还是要转为同步,因为主线程需要等待结果,故主线程需要调用CountDownLatch的await方法,进行阻塞,直到计数器为0。


关键中的关键:引入多个子线程并发执行,将同步任务转换为异步执行,但最终结果是必须等待所有子线程运行完毕,故此时异步又需要转回同步:


即CountDownLatch通过引入计数器以及countDown()方法与await()方法实现线程之间的协作,从而实现同步转异步。

相关文章
|
17天前
|
存储 Java 开发者
Java Map实战:用HashMap和TreeMap轻松解决复杂数据结构问题!
【10月更文挑战第17天】本文深入探讨了Java中HashMap和TreeMap两种Map类型的特性和应用场景。HashMap基于哈希表实现,支持高效的数据操作且允许键值为null;TreeMap基于红黑树实现,支持自然排序或自定义排序,确保元素有序。文章通过具体示例展示了两者的实战应用,帮助开发者根据实际需求选择合适的数据结构,提高开发效率。
49 2
|
23天前
|
存储 消息中间件 安全
JUC组件实战:实现RRPC(Java与硬件通过MQTT的同步通信)
【10月更文挑战第9天】本文介绍了如何利用JUC组件实现Java服务与硬件通过MQTT的同步通信(RRPC)。通过模拟MQTT通信流程,使用`LinkedBlockingQueue`作为消息队列,详细讲解了消息发送、接收及响应的同步处理机制,包括任务超时处理和内存泄漏的预防措施。文中还提供了具体的类设计和方法实现,帮助理解同步通信的内部工作原理。
JUC组件实战:实现RRPC(Java与硬件通过MQTT的同步通信)
|
9天前
|
Java 调度
Java 线程同步的四种方式,最全详解,建议收藏!
本文详细解析了Java线程同步的四种方式:synchronized关键字、ReentrantLock、原子变量和ThreadLocal,通过实例代码和对比分析,帮助你深入理解线程同步机制。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
Java 线程同步的四种方式,最全详解,建议收藏!
|
13天前
|
缓存 监控 Java
Java 线程池在高并发场景下有哪些优势和潜在问题?
Java 线程池在高并发场景下有哪些优势和潜在问题?
|
15天前
|
安全 Java 开发者
Java多线程中的`wait()`、`notify()`和`notifyAll()`方法,探讨了它们在实现线程间通信和同步中的关键作用
本文深入解析了Java多线程中的`wait()`、`notify()`和`notifyAll()`方法,探讨了它们在实现线程间通信和同步中的关键作用。通过示例代码展示了如何正确使用这些方法,并分享了最佳实践,帮助开发者避免常见陷阱,提高多线程程序的稳定性和效率。
26 1
|
20天前
|
开发框架 Java 程序员
揭开Java反射的神秘面纱:从原理到实战应用!
本文介绍了Java反射的基本概念、原理及应用场景。反射允许程序在运行时动态获取类的信息并操作其属性和方法,广泛应用于开发框架、动态代理和自定义注解等领域。通过反射,可以实现更灵活的代码设计,但也需注意其性能开销。
36 1
|
安全 Java 容器
Java并发编程 - 线程不安全类 & 同步/并发容器之简介
Java并发编程 - 线程不安全类 & 同步/并发容器之简介
112 0
Java并发编程 - 线程不安全类 & 同步/并发容器之简介
|
安全 Java 容器
java并发编程笔记3-同步容器&并发容器&闭锁&栅栏&信号量
一.同步容器:   1.Vector容器实现了List接口,Vector实际上就是一个数组,和ArrayList类似,但是Vector中的方法都是synchronized方法,即进行了同步措施。保证了线程安全。
1551 0