Springboot配置@Async无效的解决方案

 更新时间:2023年09月27日 09:57:50   作者:唐宋xy  
这篇文章主要介绍了Springboot配置@Async无效的解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教

Springboot配置@Async无效

Springboot中为了同时执行多个任务,或者同时查询,加快查询则使用多线程查询,在springboot中,可以直接使用 @Async注解来实现异步任务,可以参考Springboot使用@Async实现异步任务简单快捷使用,但是异步任务在某些情况下居然无效。

配置正确,代码不报错,但是无法使用多线程异步任务,可能是下面的原因导致:

1.没有在springboot启动类当中添加注解 @EnableAsync注解。

2.异步方法使用注解@Async的返回值只能为void或者Future。

3.没有走Spring的代理类。

因为@Transactional和@Async注解的实现都是基于Spring的AOP,而AOP的实现是基于动态代理实现的。

那么注解失效的原因就很明显了,有可能因为调用方法的是对象本身而不是代理对象,没有经过Spring容器,无法使用代理对象调用。

4.注解的方法必须是 public 方法。编译时非public方法会报错

5.如果需要从 类的内部 调用,需要先获取其代理类

先注入ApplicationContext

@Autowired
private ApplicationContext applicationContext;

然后在方法中通过ApplicationContext对象获取当前类对象

applicationContext.getBean(Class);

6.调用的是静态(static )方法,也会导致注解失效,因为注解是基于代理对象调用,而static方法是属于类的,无法通过spring的代理对象直接调用

Springboot @Async失效的坑

异步应用场景

为了提高接口的响应性能,当业务非常复杂的情况下,可以将一部分跟业务关联性不是特别强的逻辑进行异步处理。如日志记录、短信发送、增加积分等。

通常而言会将此类业务逻辑通过异步的方式进行处理,从而加快接口的响应速度,常用的解决方案有:

  • 使用JDK 自定义线程池 让代码异步执行
  • 在springboot 中 使用@Async注解进行异步处理
  • 使用中间件如mq 消息通知让下游异步消费 如RocketMQ、KAFKA

使用第一种方式,需要精通线程池运行原理,结合实际的业务场景对队列大小进行合理的设置。队列设置过大过小都会存在内存溢出的风险。

第三种方式是最合理的方式,它能够通过MQ进行削峰填谷,通过合理的参数配置,保证数据不会丢失。但是架构改动过大,对小型的单体应用来讲,工作量过大,成本过高。

在springboot 大行其道的情况下,考虑开发成本,以及项目时间关系选用第二种方式来解决代码异步执行的问题。

真实业务场景

线上问题

一个工单的分页列表,前端控制了每个列表最大的显示条数为100条。在业务流程中存在工单转移的操作,转移一笔工单至少包含以下几个重要的步骤:

  • 新增工单处理日志,如什么时间点将工单转移给某人
  • 修改工单当前处理人
  • 发送企业微信给B端的跟进人(转移人)
  • 发送im信息给C端的用户

由于公司采用微服务架构,因此每个业务模块拆分的很细,在上述步骤中需要从其他系统中通过rpc调用接口拿到需要的数据才能完成整个业务流程数据的拼装,如需要从crm系统拿到组织架构信息,获取转移人的组织架构、需要从udb用户中心获取转移人的企业微信昵称等。

因此在批量转移的时候,前端会出现调用超时的问题,原因是dubbo接口默认的超时时间是15秒,由于业务复杂,导致在15秒内执行不完业务逻辑。

解决方案

  • 将sql处理改为批量执行,如新增处理日志 (batch insert);修改当前工单,使用case when 的方式一次性修改完成(批量update)
  • 将发送消息改成异步处理 加快前端接口的响应速度
  • 让接口提供方提供批量查询的接口,避免rpc 循环调用在网络上的消耗

优化完成之后,接口的响应速度由15秒多,变成了1秒。但是过程中遇到坑了。特此记录一下

技术实现

优化过程

@Async 注解定义为可以放置在方法上和类上,当使用在类上表明类所有的方法都能异步执行。

在Springboot中是需要在方法上加上该注解就可以完美的实现异步执行。

原始方法伪代码如下

/**
 * 原始代码 采用流水式的代码一步步实现 业务逻辑
 */
public void doBusiness(Object args){
  //1.  新增工单处理日志,如什么时间点将工单转移给某人
  //2. 修改工单当前处理人
  //3. 发送企业微信给B端的跟进人(转移人)
  //4. 发送im信息给C端的用户
}

那么异步问题就很好处理了,只需要将方法抽离形成多个子方法, 每个方法执行自己的业务处理逻辑,然后再方法加上@Async注解不就ok了么,伪代码如下

public void doBusiness(Object args){
 	//2. 修改工单当前处理人
  this.doAsyncBusiness();
}
// 单独抽离一个异步执行的方法 加上@Async注解
@Async
private void doAsyncBusiness(Object args){
  //1.  新增工单处理日志,如什么时间点将工单转移给某人
  //3. 发送企业微信给B端的跟进人(转移人)
  //4. 发送im信息给C端的用户
}

打完收工,重启应用,进行测试,然而并没有像预期中的那样,接口的响应速度还是15秒左右。接着排查原因,可以肯定的是@Async是可以提供异步方法执行。应该是我们使用方式不对导致。

@Async 限制

熟悉Springboot AOP的同学可能会发现更改后的代码存在明显的问题

  • 首先AOP代理机制要求 被代理的方法必须是 public , private 方法不能被代理
  • 其次AOP代理机制会生成一个代理类 执行代理方法 注意 this.doAsyncBusiness() 调用的是本对象的方法 ;
  • 在启动类上加上@EnableAsync注解

综上所述,原因我们已经通过AOP代理的原理找到了。下面摘自官方文档的一段话:

  • it must be applied to public methods only
  • self-invocation – calling the async method from within the same class – won’t work

The reasons are simple – 「the method needs to be public」 so that it can be proxied. And 「self-invocation doesn’t work」 because it bypasses the proxy and calls the underlying method directly.

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • 利用Intellij Idea连接远程服务器实现远程上传部署功能

    利用Intellij Idea连接远程服务器实现远程上传部署功能

    大家在使用Intellij Idea开发程序的时候,是不是需要部署到远程SSH服务器运行呢,当然也可以直接在idea软件内容实现配置部署操作,接下来通过本文给大家分享利用Intellij Idea连接远程服务器实现远程上传部署功能,感兴趣的朋友跟随小编一起看看吧
    2021-05-05
  • 详解Spring 注解之@Import 注入的各种花活

    详解Spring 注解之@Import 注入的各种花活

    这篇文章主要介绍了详解Spring 注解之@Import 注入的各种花活,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-01-01
  • Java变量与运算符用法详解

    Java变量与运算符用法详解

    这篇文章介绍了Java中的关键字、标识符、变量、基本数据类型、变量作用域、基本数据类型间的运算、字符串、基本数据类型与字符串的运算、计算机底层数据存储、运算符的使用、赋值运算符、比较运算符、逻辑运算符、位运算符和条件运算符用法,感兴趣的朋友跟随小编看看吧
    2026-01-01
  • Java之Thread的join方法实例

    Java之Thread的join方法实例

    这篇文章主要介绍了Java之Thread的join方法,实例形式讲述了join方法的应用,需要的朋友可以参考下
    2014-10-10
  • RepeatSubmit若依框架如何防止表单重复提交注解

    RepeatSubmit若依框架如何防止表单重复提交注解

    若依框架中的@RepeatSubmit注解用于防止表单重复提交,通过在控制器方法上添加该注解,并在前端页面和JavaScript代码中实现双重校验,可以确保同一用户在短时间内不会重复提交相同的表单
    2024-11-11
  • 用java生成html文件实现原理及代码

    用java生成html文件实现原理及代码

    用printStream来向html文件里输出数据,先创建一个StringBuilder对象,通过append方法来为其添加html语句,具体实现如下,感兴趣的朋友可以参考下,希望对大家有所帮助
    2013-08-08
  • springboot springmvc抛出全局异常的解决方法

    springboot springmvc抛出全局异常的解决方法

    这篇文章主要为大家详细介绍了springboot springmvc抛出全局异常的解决方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-06-06
  • Java的Hibernate框架中的双向主键关联与双向外键关联

    Java的Hibernate框架中的双向主键关联与双向外键关联

    Hibernate想要实现双向的关联就必须在映射文件的两端同时配置<one-to-one>,另外还要在主映射的一端采用foreign外键关联属性,下面我们就一起来看一下Java的Hibernate框架中的双向主键关联与双向外键关联方法:
    2016-06-06
  • 浅谈Java响应式系统

    浅谈Java响应式系统

    第一次听到reactive这个词还是在几年前,偶然了解到了Rxjava这个项目,仿佛为我打开了一扇新的大门,Rxjava是ReactiveX的java实现,ReactiveX家族除了Rxjava还有RxJS, Rx.NET,RxScala等等。
    2021-06-06
  • Java设计模式之责任链模式

    Java设计模式之责任链模式

    今天小编就为大家分享一篇关于Java设计模式之责任链模式,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-01-01

最新评论