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

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);});
}

相关文章:

  • Leetcode 269. 火星词典
  • Python----目标检测(《SSD: Single Shot MultiBox Detector》论文和SSD的原理与网络结构)
  • Redis:安装与常用命令
  • 基于ZYNQ ARM+FPGA异构平台的声呐数据采集系统设计
  • ARM P15协处理器指令详解:架构、编程与应用实践
  • 支持向量机(SVM):解锁数据分类与回归的强大工具
  • C#语音识别:使用Whisper.net实现语音识别
  • PySide6 GUI 学习笔记——常用类及控件使用方法(标签控件QLabel)
  • 鸿蒙OS基于UniApp的区块链钱包开发实践:打造支持鸿蒙生态的Web3应用#三方框架 #Uniapp
  • LeetCode 热题 100 155. 最小栈
  • unix/linux source 命令,其高级使用
  • 历史数据分析——广州港
  • C#里与嵌入式系统W5500网络通讯(3)
  • Java补充(Java8新特性)(和IO都很重要)
  • 零基础安装 Python 教程:从下载到环境配置一步到位(支持 VSCode 和 PyCharm)与常用操作系统操作指南
  • 【Go-6】数据结构与集合
  • 【Java】JDK 命令行工具
  • WIN11+VSCODE搭建的c/c++环境调试报错解决
  • 数据要素×AI:高质量数据集如何成为智能时代的“新石油“
  • Python 中 dpkt 库的详细使用指南(强大的 Python 数据包解析库)
  • 做网店的进货网站/制作网页
  • 城阳建网站/seo首页优化
  • 二手手机回收网站开发/项目外包平台
  • 做网站还需要搜狗吗/微信营销系统
  • 网站更新要怎么做/推广接单平台
  • 做自媒体要知道的网站/软文写作范文