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

springboot中@Async做异步操作(Completable异步+ThreadPoolTaskExecutor线程池+@Async注解)

文章目录

  • 前置
  • 启用异步支持
  • 编写异步方法
  • 关于线程池(强烈建议自定义线程池!)
  • 为什么 @Async 异步未生效?
  • 异步方法的异常怎么处理
  • 异步方法的返回值
  • 使用场景举例子

前置

@Async用于实现方法级别的异步执行,本质是将方法调用交给后台线程池执行,从而避免阻塞主线程,提升系统的吞吐量和响应速度

强烈建议使用 @Async 时候,需要自定义线程池!

  • 如果是一般 java 项目,现在更多是使用 CompletableFuture 配合上 ExecutorService 开启异步操作
  • 如果是 java springboot web 项目,更多的是使用 @Async 配合上 ThreadPoolTaskExecutor 配置线程池和结合 CompletableFuture 一起做异步操作

启用异步支持

要使用 @Async,首先需要在 Spring 配置类或启动类上添加 @EnableAsync 注解,以启用异步功能
@EnableAsync // 启用异步支持

编写异步方法

在需要异步执行的方法上添加 @Async 注解,该方法必须是 spring 管理的 bean 中的方法,方法也不能是 final,private 这种

import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Service;@Service
public class EmailService {@Async  // 标记为异步方法public void sendEmail(String emailContent) {try {// 模拟发送邮件耗时操作Thread.sleep(2000);System.out.println("邮件发送完成,内容:" + emailContent);} catch (InterruptedException e) {e.printStackTrace();}}
}

关于线程池(强烈建议自定义线程池!)

默认情况下,@Async 使用 Spring 的 SimpleAsyncTaskExecutor 线程池(最大线程数为 Integer.MAX_VALUE),一旦请求激增,有可能会导致 cpu 陡增和内存陡增,导致服务崩溃

强烈建议自定义线程池!

import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;@Configuration
@EnableAsync
public class AsyncConfig {@Bean(name = "customTaskExecutor")public ThreadPoolTaskExecutor customTaskExecutor() {ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();executor.setCorePoolSize(5);          // 核心线程数executor.setMaxPoolSize(10);           // 最大线程数executor.setQueueCapacity(100);        // 队列容量,避免无边界队列executor.setThreadNamePrefix("Async-"); // 线程名称前缀executor.initialize();return executor;}
}

在 @Async 注解中通过 value 参数指定线程池名称:

@Async("customTaskExecutor")  // 使用自定义线程池
public void sendEmail(String emailContent) {// ...
}

为什么 @Async 异步未生效?

  • 未启用 @EnableAsync:检查启动类或配置类是否缺少 @EnableAsync
  • 如果方法在同一个类中被调用,@Async 会失效(Spring 通过代理实现异步,内部调用不走代理)
  • @Async 方法不能是 private 或 final

异步方法的异常怎么处理

异步方法中的异常不会传播到调用方,建议在方法内部捕获并记录

@Async
public void sendEmail(String content) {try {// 业务逻辑} catch (Exception e) {// 记录日志System.err.println("异步任务异常: " + e.getMessage());}
}

异步方法的返回值

@Async 方法可以返回 void 或 Future<T>
Future.get()能获取结果,但是会阻塞主线程

使用场景举例子

常结合 CompletableFuture 使用

@Async
public CompletableFuture<User> getUserAsync(Long userId) {// 调用用户服务
}@Async
public CompletableFuture<Order> getOrderAsync(Long userId) {// 调用订单服务
}// 聚合结果
public void aggregateData(Long userId) {CompletableFuture<User> userFuture = getUserAsync(userId);CompletableFuture<Order> orderFuture = getOrderAsync(userId);userFuture.thenCombine(orderFuture, (user, order) -> {// 处理聚合结果return new UserProfile(user, order);});
}
http://www.xdnf.cn/news/10529.html

相关文章:

  • shp转3d tiles在cesium渲染楼宇白膜
  • Linux 驱动之设备树
  • Leetcode 2093. 前往目标城市的最小费用
  • SAR ADC 异步逻辑设计
  • Linux系统配置屏幕旋转和触摸旋转
  • 从冷上电到main()函数,Bootloader都做了什么?
  • 数据类型检测有哪些方式?
  • robot_lab学习笔记【MDP综述】
  • QuickJS 如何计算黄金分割率 ?
  • barker-OFDM模糊函数原理及仿真
  • Linux防火墙:全面解析IPTables的表、链、规则!
  • Cypress + TypeScript + Vue3
  • 数据库管理与高可用-MySQL全量,增量备份与恢复
  • 劫持进程注入
  • C语言进阶--程序的编译(预处理动作)+链接
  • 数据结构:递归(Recursion)
  • 基于TMC5160堵转检测技术的夹紧力控制系统设计与实现
  • 输入ifconfig,发现ens33不见了,无法连接至虚拟机
  • Golang——3、流程控制语句
  • C++实现伽罗华域生成及四则运算(三)
  • Python----目标检测(《SSD: Single Shot MultiBox Detector》论文和SSD的原理与网络结构)
  • CppCon 2014 学习:C++ in Huge AAA Games
  • STM32F407寄存器操作(多通道单ADC+DMA)
  • 前端面试准备-5
  • Mask_RCNN 环境配置及训练
  • QT中子线程触发主线程弹窗并阻塞等待用户响应-传统信号槽实现
  • DRW - 加密市场预测
  • 考研系列—操作系统:第四章、文件管理(part.2)
  • 利用DeepSeek编写能在DuckDB中读PostgreSQL表的表函数
  • 多任务——进程