Java——异步调用
2021/9/1 11:06:22
本文主要是介绍Java——异步调用,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!
一、通过创建新线程
异步调用的本质,其实是通过开启一个新的线程来执行。如以下例子:
public static void main(String[] args) throws Exception{ System.out.println("主线程 =====> 开始 =====> " + System.currentTimeMillis()); new Thread(() -> { System.out.println("异步线程 =====> 开始 =====> " + System.currentTimeMillis()); try{ Thread.sleep(5000); }catch (InterruptedException e){ e.printStackTrace(); } System.out.println("异步线程 =====> 结束 =====> " + System.currentTimeMillis()); }).start(); Thread.sleep(2000); System.out.println("主线程 =====> 结束 =====> " + System.currentTimeMillis()); }
数据结果如下所示:
主线程 =====> 开始 =====> 1627893837146 异步线程 =====> 开始 =====> 1627893837200 主线程 =====> 结束 =====> 1627893839205 异步线程 =====> 结束 =====> 1627893842212
二、通过线程池
因为异步任务的实现本质的由新线程来执行任务,所以通过线程池的也可以实现异步执行。写法同我们利用线程池开启多线程一样。
但由于我们的目的不是执行多线程,而是异步执行任务,所以一般需要另外一个线程就够了。
因此区别于执行多线程任务的我们常用的newFixedThreadPool
,在执行异步任务时,我们用newSingleThreadExecutor
来创建一个单个线程的线程池。
public static void main(String[] args) throws Exception{ System.out.println("主线程 =====> 开始 =====> " + System.currentTimeMillis()); ExecutorService executorService = Executors.newSingleThreadExecutor(); executorService.submit(()->{ System.out.println("异步线程 =====> 开始 =====> " + System.currentTimeMillis()); try{ Thread.sleep(5000); }catch (InterruptedException e){ e.printStackTrace(); } System.out.println("异步线程 =====> 结束 =====> " + System.currentTimeMillis()); }); executorService.shutdown(); // 回收线程池 Thread.sleep(2000); System.out.println("主线程 =====> 结束 =====> " + System.currentTimeMillis()); }
执行结果如下:
主线程 =====> 开始 =====> 1627895467578 异步线程 =====> 开始 =====> 1627895467635 主线程 =====> 结束 =====> 1627895469644 异步线程 =====> 结束 =====> 1627895472649
三、通过@Async注解
SpringBoot项目有一个的很重要的特点就是的注解化。如果你的项目是SpringBoot,那就又多了一种选择——@Async注解。
使用起来也非常简单,将要异步执行的代码封装成一个方法,然后用@Async注解该方法,然后在主方法中直接调用就行。
@Test public void mainThread() throws Exception{ System.out.println("主线程 =====> 开始 =====> " + System.currentTimeMillis()); collectionBill.asyncThread(); Thread.sleep(2000); System.out.println("主线程 =====> 结束 =====> " + System.currentTimeMillis()); Thread.sleep(4000); // 用于防止jvm停止,导致异步线程中断 }
@Async public void asyncThread(){ System.out.println("异步线程 =====> 开始 =====> " + System.currentTimeMillis()); try{ Thread.sleep(5000); }catch (InterruptedException e){ e.printStackTrace(); } System.out.println("异步线程 =====> 结束 =====> " + System.currentTimeMillis()); }
执行结果如下:
主线程 =====> 开始 =====> 1627897539948 异步线程 =====> 开始 =====> 1627897539956 主线程 =====> 结束 =====> 1627897541965 异步线程 =====> 结束 =====> 1627897544966
有以下两点需要注意:
- 类似
@Tranctional
注解,@Async
注解的方法与调用方法不能在同一个类中,否则不生效 - JUnit框架的设计不考虑多线程场景,所以主线程退出后,子线程也会跟着立即退出,所以可以在后面加多线程休眠时间来观察异步线程的执行情况
四、通过CompletableFuture
CompletableFuture是JDK1.8的新特性,是对Future的扩展。CompletableFuture实现了CompletionStage接口和Future接口,增加了异步回调、流式处理、多个Future组合处理的能力。
实现代码如下:
public static void main(String[] args) throws Exception{ System.out.println("主线程 =====> 开始 =====> " + System.currentTimeMillis()); ExecutorService executorService = Executors.newSingleThreadExecutor(); CompletableFuture.runAsync(() ->{ System.out.println("异步线程 =====> 开始 =====> " + System.currentTimeMillis()); try{ Thread.sleep(5000); }catch (InterruptedException e){ e.printStackTrace(); } System.out.println("异步线程 =====> 结束 =====> " + System.currentTimeMillis()); },executorService); executorService.shutdown(); // 回收线程池 Thread.sleep(2000); System.out.println("主线程 =====> 结束 =====> " + System.currentTimeMillis()); }
同样可以实现类似的结果如下:
主线程 =====> 开始 =====> 1627898354914 异步线程 =====> 开始 =====> 1627898354977 主线程 =====> 结束 =====> 1627898356980 异步线程 =====> 结束 =====> 1627898359979
五、通过MQ
通过消息队列中间件实现异步处理,也是很常见的一种方式,比如:RabbitMQ、RocketMQ等,这里就不举例了。
六、通过事件处理机制
我们项目中,某些场景下使用的是Google的 EventBus。EventBus 是 Google.Guava 的事件处理机制,是观察者模式(生产/消费模型)的一种实现。
类似的还有Spring的Event。
EventBus 优点:
- 相比 Observer 编程简单方便
- 通过自定义参数可实现同步、异步操作以及异常处理
- 单进程使用,无网络影响
EventBus 缺点:
- 只能单进程使用
- 项目异常重启或者退出不保证消息持久化
如果需要分布式使用还是需要使用 MQ。
具体实现请参考:https://juejin.cn/post/6864940197269667853
这篇关于Java——异步调用的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!
- 2025-01-14折腾之王:JavaScript 之父 Brave 浏览器与 BAT 的诞生
- 2025-01-13从协作到创新:电商团队效率提升新方法
- 2025-01-13汉服销售拓展客源,能精准投放广告的软件求推荐!蛇年新春!
- 2025-01-13提升客户体验的关键:电商团队协作效率优化
- 2025-01-13不触碰资金的支付网关有哪些?
- 2025-01-13如何运用敏捷开发的6大模型来提高团队工作效率?
- 2025-01-13汉服制作质量检测,能高清放大细节的软件用哪个?2025 新春!
- 2025-01-13团队目标管理的6种实用方法(附OKR模板)
- 2025-01-13短剧制片的幕后:如何通过协同编辑提升效率
- 2025-01-13B端产品业务调研:系统步骤与策略解析