SpringBoot使用Async注解失效原因分析及解决(spring异步回调)

 更新时间:2021年10月18日 11:11:35   作者:u011277123  
这篇文章主要介绍了SpringBoot使用Async注解失效原因分析及解决(spring异步回调),具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

Async注解失效原因分析及解决(spring异步回调)

Spring中@Async

在Java应用中,绝大多数情况下都是通过同步的方式来实现交互处理的;但是在处理与第三方系统交互的时候,容易造成响应迟缓的情况,之前大部分都是使用多线程来完成此类任务,其实,在spring 3.x之后,就已经内置了@Async来完美解决这个问题

有时候在使用的过程中@Async注解会失效

(原因和@Transactional注解有时候会失效的原因一样)。

下面定义一个Service:

两个异步执行的方法test03()和test02()用来模拟项目中可能出现的耗时操作,然后test()方法调用这两个耗时的方法:

SpringBoot使用Async注解失效分析、解决(spring异步回调)

定义Controller:

SpringBoot使用Async注解失效分析、解决(spring异步回调)

执行方法,返回结果:

SpringBoot使用Async注解失效分析、解决(spring异步回调)

方法执行结果明显与我们的预期不符,方法的输出顺序表示了test02()和test03()两个异步方法居然同步执行了,也就是说@Aysnc注解失效了!

失效的原因是因为我们是在test()方法中直接调用的test02()和test03()方法,相当于是this.test02()和this.test03()调用的,也就是说真正调用test02()和test03()方法的是TestService对象本身调用的,而@Async和@Transactional注解本质使用的是动态代理,真正应该是TestService的代理对象调用test02()和test03()方法。其实Spring容器在初始化的时候Spring容器会将含有AOP注解的类对象“替换”为代理对象(简单这么理解),那么注解失效的原因就很明显了,就是因为调用方法的是对象本身而不是代理对象,因为没有经过Spring容器,那么解决方法也会沿着这个思路来解决。

网上有不少博客说解决方法就是将要异步执行的方法单独抽取成一个类,这样的确可以解决异步注解失效的问题,原理就是当你把执行异步的方法单独抽取成一个类的时候,这个类肯定是被Spring管理的,其他Spring组件需要调用的时候肯定会注入进去,这时候实际上注入进去的就是代理类了,其实还有其他的解决方法,并不一定非要单独抽取成一个类。

解决方式一

在TestService中通过上下文获取自己的代理对象调用异步方法

其实我们的注入对象都是从Spring容器中给当前Spring组件进行成员变量的赋值,由于TestService使用了AOP注解,那么实际上TestService在Spring容器中实际存在的是它的代理对象。

SpringBoot使用Async注解失效分析、解决(spring异步回调)

执行结果,异步方法异步执行了:

SpringBoot使用Async注解失效分析、解决(spring异步回调)

解决方式二

开启cglib代理,手动获取Spring代理类

在启动类上加上:

SpringBoot使用Async注解失效分析、解决(spring异步回调)

使用AopContext.currentProxy()获取当前代理类:

这里为了证明Spring容器中的对象就是当前代理类对象特地输出了一句话:

SpringBoot使用Async注解失效分析、解决(spring异步回调)

运行结果:

SpringBoot使用Async注解失效分析、解决(spring异步回调)

OK,问题完美解决!

application.properties配置如下:

#java.lang.IllegalStateException: Cannot find current proxy: Set 'exposeProxy' property on Advised to 'true' to make it available.
# 增加@EnableAspectJAutoProxy
spring.aop.auto=true
#开启CGLIB代理
spring.aop.proxy-target-class=true

springboot @Async 失效可能原因

1、当前类中其他函数调用有 @Async 注解的函数

2、当前类中有多态,方法名相同

3、启动类未加@EnableAsync

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

相关文章

  • IDEA如何对单个的java class文件打成jar包

    IDEA如何对单个的java class文件打成jar包

    这篇文章主要介绍了IDEA如何对单个的java class文件打成jar包问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-08-08
  • springboot解决XSS存储型漏洞问题

    springboot解决XSS存储型漏洞问题

    这篇文章主要介绍了springboot解决XSS存储型漏洞问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-05-05
  • SpringBoot实现自定义条件注解的代码示例

    SpringBoot实现自定义条件注解的代码示例

    在Spring Boot中,条件注解是一种非常强大的工具,它可以根据特定的条件来选择是否加载某个类或某个Bean,文将介绍如何在Spring Boot中实现自定义条件注解,并提供一个示例代码,需要的朋友可以参考下
    2023-06-06
  • Java实现红黑树(平衡二叉树)的详细过程

    Java实现红黑树(平衡二叉树)的详细过程

    红黑树接近平衡的二叉树,插入,删除函数跟平衡二叉树一样,只是平衡函数不同,下面这篇文章主要给大家介绍了关于Java实现红黑树(平衡二叉树)的相关资料,需要的朋友可以参考下
    2021-10-10
  • Java使用easypoi快速导入导出的实现

    Java使用easypoi快速导入导出的实现

    这篇文章主要介绍了实现Java使用easypoi快速导入导出的实现,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2019-03-03
  • Mybatis Plus使用@TableId的示例详解

    Mybatis Plus使用@TableId的示例详解

    在 MyBatis Plus 中,@TableId 注解是用于标记实体类中的主键字段,它可以更方便地处理主键相关的操作,如自动填充主键值或识别主键字段,这篇文章主要介绍了Mybatis Plus使用@TableId,需要的朋友可以参考下
    2024-08-08
  • SpringBoot实现其他普通类调用Spring管理的Service,dao等bean

    SpringBoot实现其他普通类调用Spring管理的Service,dao等bean

    这篇文章主要介绍了SpringBoot实现其他普通类调用Spring管理的Service,dao等bean,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-11-11
  • Spring Cloud Gateway 拦截响应问题分析(数据截断问题)

    Spring Cloud Gateway 拦截响应问题分析(数据截断问题)

    这篇文章主要介绍了Spring Cloud Gateway 拦截响应问题分析(数据截断问题),本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-01-01
  • SpringBoot使用WebSocket的方法实例详解

    SpringBoot使用WebSocket的方法实例详解

    WebSocket 是 HTML5 开始提供的一种在单个 TCP 连接上进行全双工通讯的协议。这篇文章主要介绍了SpringBoot使用WebSocket,需要的朋友可以参考下
    2019-06-06
  • springboot Long 精度丢失问题解决

    springboot Long 精度丢失问题解决

    这篇文章主要为大家介绍了解决springboot Long 精度丢失问题的方法详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-07-07

最新评论