Springboot @Async 异步调用的使用方法与注意事项

本文深入探讨SpringBoot中异步调用@Async的使用方法,包括配置、常见问题及解决方案,如事务处理、方法定义位置和异步方法必须实例化的原因。并提供代码示例帮助理解。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

异步调用的概念
异步调用相对于同步调用而言,通常的方法都是程序按照顺序来执行的,程序的每一步都需要等到上一步执行完成之后才能继续往下执行;而异步调用则无需等待,它可以在不阻塞主线程的情况下执行高耗时方法

如何实现异步调用
在不使用SpringBoot的时候我们通常使用多线程的方式来实现异步调用,对于一个web项目的话就需要使用线程池来创建多线程进行调用,而SpringBoot提供了很方便的@Async注解来实现异步方法

需要引入的jar包:

 
  1. <parent>  

  2.     <groupId>org.springframework.boot</groupId>  

  3.     <artifactId>spring-boot-starter-parent</artifactId>  

  4.     <version>1.5.3.RELEASE</version>  

  5. </parent>  

  6. <dependencies>  

  7.     <dependency>  

  8.         <groupId>org.springframework.boot</groupId>  

  9.         <artifactId>spring-boot-starter-web</artifactId>  

  10.     </dependency>  

  11. </dependencies>  


需要使用到的注解:
@Async:用于指定注解的方式为异步调用的方法 
@EnableAsync:用于开启异步调用机制,注解在有@Config注解修饰的类上

需要注意的问题一:异步方法的定义位置问题
最好将异步调用的方法单独放在一个@Component类中,或者说不要将异步调用方法写在@Controller中,否则将无法进行调用,因为SpringBoot使用@Transaction需要经过事务拦截器,只有通过了该事务拦截器的方法才能被加入Spring的事务管理器中,而在同一个类中的一个方法调用另一个方法只会经过一次事务拦截器,所以如果是后面的方法使用了事务注解将不会生效,在这里异步调用也是同样的道理

参考文献
spring事务@Transactional在同一个类中的方法调用不生效 
SpringBoot集成篇(二) 异步调用Async

需要注意的问题二:异步方法的事务调用问题
在@Async注解的方法上再使用@Transaction注解是无效的,在@Async注解的方法中调用Service层的事务方法是有效的

需要注意的问题三:异步方法必须是实例的
因为静态方法不能被override重写,因为@Async异步方法的实现原理是通过注入一个代理类到Bean中,该代理类集成这个Bean并且需要重写这个异步方法,所以需要是实例方法

参考文献
Springboot @Async 异步方法

代码示例
可以参考该文章里面的SpringBoot集成篇(二) 异步调用Async


--------------------- 
作者:19年,重新开始写 
来源:优快云 
原文:https://blog.youkuaiyun.com/qq_27905183/article/details/80294497 
版权声明:本文为博主原创文章,转载请附上博文链接!

### SpringBoot 中 `@Async` 注解的配置方法和详细说明 #### 一、启用异步支持 为了在 Spring Boot 应用中使用 `@Async` 实现异步调用,需要通过在主类或配置类上添加 `@EnableAsync` 注解来启用异步功能。以下是具体的代码示例: ```java import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.scheduling.annotation.EnableAsync; @SpringBootApplication @EnableAsync public class AsyncApplication { public static void main(String[] args) { SpringApplication.run(AsyncApplication.class, args); } } ``` 此操作会激活 Spring 的异步执行机制[^1]。 --- #### 二、定义异步方法 当启用了异步支持后,在业务逻辑层可以通过标注 `@Async` 来声明某个方法为异步方法。被标记的方法会在单独的线程中运行,而不会阻塞主线程。以下是一个简单的例子: ```java import org.springframework.scheduling.annotation.Async; import org.springframework.stereotype.Service; @Service public class MyAsyncService { @Async public void performTask() { try { Thread.sleep(2000); // 模拟耗时任务 } catch (InterruptedException e) { throw new RuntimeException(e); } System.out.println("异步任务完成"); } } ``` 需要注意的是,默认情况下,Spring 使用单一线程池(SimpleAsyncTaskExecutor)来管理这些异步任务。如果项目中有大量并发请求,则可能需要自定义线程池以优化性能[^2]。 --- #### 三、自定义线程池配置 对于高负载的应用场景,建议创建并注入一个更高效的线程池替代默认实现。下面展示如何基于 `ThreadPoolTaskExecutor` 自定义线程池: ```java package com.example.asyncdemo.config; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor; @Configuration public class AsyncConfig { private static final int MAX_POOL_SIZE = 50; private static final int CORE_POOL_SIZE = 20; @Bean(name = "customAsyncTaskExecutor") public ThreadPoolTaskExecutor customAsyncTaskExecutor() { ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor(); executor.setCorePoolSize(CORE_POOL_SIZE); executor.setMaxPoolSize(MAX_POOL_SIZE); executor.setQueueCapacity(100); executor.setThreadNamePrefix("Custom-Async-Thread-"); executor.initialize(); return executor; } } ``` 在此基础上,可以指定特定的任务由该线程池处理。只需修改目标服务中的 `@Async` 方法签名即可绑定到新线程池: ```java import org.springframework.beans.factory.annotation.Autowired; import org.springframework.scheduling.annotation.Async; import org.springframework.stereotype.Service; @Service public class CustomMyAsyncService { @Autowired private ThreadPoolTaskExecutor customAsyncTaskExecutor; @Async("customAsyncTaskExecutor") // 显式指定了使用的线程池名称 public void performCustomTask() { try { Thread.sleep(3000); // 模拟耗时任务 } catch (InterruptedException e) { throw new RuntimeException(e); } System.out.println("定制化异步任务完成"); } } ``` 以上方式允许开发者灵活控制不同类型的异步任务所占用资源的比例[^3]。 --- #### 四、注意事项 1. **返回值类型** 被 `@Async` 标记的方法通常有两种合法的返回值形式:`void` 或者 `Future<T>` 类型及其子类(如 CompletableFuture)。如果是后者,则可以在调用方获取结果的同时保持非阻塞性质。 2. **事务传播行为** 如果在一个已有的事务上下文中调用了一个带有 `@Transactional` 和 `@Async` 双重修饰的服务函数,那么新的事务将会独立启动而不是继承父级事务的状态[^4]。 3. **异常捕获** 异常发生在不同的线程里头,因此常规手段无法直接感知远程抛出的问题。推荐采用回调监听器或者 AOP 技术来进行全局错误记录反馈。 --- #### 总结 综上所述,借助于 `@Async` 结合适当规模大小调整后的线程池方案,能够有效提升应用程序面对复杂计算密集型作业时的整体吞吐量表现。同时也要注意合理规划各组件间协作关系以免引入不必要的同步开销影响最终效果。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值