当前位置: 首页 > news >正文

异步调用实践:Async,Future, TaskExecutor、EventListener

1. 异步调用概述

异步调用允许一个方法调用在不被当前线程阻塞的情况下继续执行,而调用者可以继续执行其他任务,直到异步操作完成。

在Spring Boot中,异步调用常用于提高应用的响应性和吞吐量,尤其是在处理长时间运行的任务时,如文件处理、数据库查询、网络请求等。

2. 异步调用注意点

  • 线程安全性:确保异步方法中访问的共享资源是线程安全的。
  • 异常处理:异步方法中的异常需要被捕获并适当处理,或者通过某种机制返回给调用者。
  • 返回类型:异步方法的返回类型通常是voidFuture<T>CompletableFuture<T>,以便调用者可以检查异步操作的结果或状态。
  • 配置异步执行器:根据需要配置线程池大小等参数,以避免资源耗尽。
  • 避免在相同类的方法中调用异步方法:因为这将绕过Spring的代理机制,导致异步调用不生效。

3. 异步调用场景

假设我们有一个在线书店应用,用户下单后需要发送订单确认邮件和更新库存。

这两个操作都是耗时的,我们希望它们异步执行,以便用户能够立即得到订单提交的反馈。

4. 具体实现

(1)@Async

首先,需要在Spring Boot中启用异步支持,在启动类上添加@EnableAsync注解。然后,在需要异步执行的方法上添加@Async注解。

@SpringBootApplication
@EnableAsync
public class BookstoreApplication {public static void main(String[] args) {SpringApplication.run(BookstoreApplication.class, args);}
}@Service
public class OrderService {@Asyncpublic void processOrder(Order order) {// 模拟发送邮件sendOrderConfirmationEmail(order);// 模拟更新库存updateStock(order);}// 实现sendOrderConfirmationEmail和updateStock方法
}

(2)CompletableFuture

CompletableFuture提供了更灵活的异步编程模型,允许你编写非阻塞代码,同时以声明方式处理异步完成时的结果。

@Service
public class OrderService {public CompletableFuture<Void> processOrderAsync(Order order) {CompletableFuture<Void> emailFuture = CompletableFuture.runAsync(() -> sendOrderConfirmationEmail(order));CompletableFuture<Void> stockFuture = CompletableFuture.runAsync(() -> updateStock(order));return CompletableFuture.allOf(emailFuture, stockFuture).thenApply(v -> null);}// 实现sendOrderConfirmationEmail和updateStock方法
}

(3)TaskExecutor

TaskExecutor是Spring提供的用于执行异步任务的接口。你可以通过配置一个TaskExecutor来管理线程池。

@Configuration
@EnableAsync
public class AsyncConfig implements AsyncConfigurer {@Overridepublic Executor getAsyncExecutor() {ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();executor.setCorePoolSize(5);executor.setMaxPoolSize(10);executor.setQueueCapacity(25);executor.initialize();return executor;}}@Service
public class OrderService {@Autowiredprivate TaskExecutor taskExecutor;public void processOrder(Order order) {taskExecutor.execute(() -> sendOrderConfirmationEmail(order));taskExecutor.execute(() -> updateStock(order));}// 实现sendOrderConfirmationEmail和updateStock方法
}

(4)@EventListener

虽然@EventListener主要用于事件监听,但你可以结合异步任务来监听特定事件(如订单创建事件),并异步处理。

@Component
public class OrderEventListener {@Async@EventListenerpublic void handleOrderCreatedEvent(OrderCreatedEvent event) {Order order = event.getOrder();sendOrderConfirmationEmail(order);updateStock(order);}// 实现sendOrderConfirmationEmail和updateStock方法
}// 假设你有一个OrderCreatedEvent类
public class OrderCreatedEvent extends ApplicationEvent {private Order order;public OrderCreatedEvent(Object source, Order order) {super(source);this.order = order;}// getter和setter
}

5. 总结

在Spring Boot中,实现异步调用可以通过多种方式,包括@Async注解、CompletableFuture、自定义TaskExecutor以及结合@EventListener进行异步事件处理。

每种方式都有其适用场景和优缺点,开发者应根据实际需求选择最合适的方法。

同时,使用异步调用时需要注意线程安全、异常处理以及合理配置异步。

http://www.lryc.cn/news/421561.html

相关文章:

  • Flask 异常处理
  • 【海思SS626 | 内存管理】海思芯片的OS内存、MMZ内存设置
  • linux crontab没有按照规则执行排查
  • Cloudflare的D1使用技巧
  • 解决端口号被占用问题
  • 如何在linux上部署zabbix监控工具
  • vulnhub系列:sp eric
  • JVM二:JVM类加载机制
  • 对于springboot无法连接redis解决方案
  • 关于android中的各种尺寸与计算
  • MySQL避免索引失效的方法详细介绍
  • 【Java】深入了解 Java 的 charAt() 方法
  • Linux 下 ETCD 安装、配置与命令使用总结
  • C++笔试练习笔记【7】:力扣 91. 解码方法 动态规划练习
  • 【antd】antd3的表单校验不提示报错信息
  • Game AI ——游戏人工智能(逻辑及剧情生成)
  • 算法基础知识——核函数
  • 安卓xml乱码/加密转换:abx2xml和xml2abx使用及源码介绍
  • slice 截取
  • XReparentWindow踩坑分析
  • OpenAI动荡,将走向何方、GPT5或许将近、毒舌AI轻松破防网友、最新版 GPT-4o AI 模型得满分 | AGI视界周刊第 4 期
  • RCE---无字母数字webshell
  • 有意思的漏洞复现与分析一
  • 力扣题解(按身高排序)
  • Redis的六种淘汰策略详解
  • vue3中 ref 和 reactive 的区别
  • 《单例模式的深度解读:实现方式、破坏情况与利弊权衡》
  • 010607电压源和电流源受控源
  • 快乐数求解
  • 运维高级内容--为端口做标记、制定调度规则