博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
SpringBoot使用Async注解失效分析、解决(spring异步回调)
阅读量:4293 次
发布时间:2019-05-27

本文共 1526 字,大约阅读时间需要 5 分钟。

原创 专注JavaWeb开发 2018-12-24 17:30:33

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容器中实际存在的是它的代理对象。

SpringUtil工具类可以参考:http://mp.toutiao.com/preview_article/?pgc_id=6638488982025929223

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.# 增加@EnableAspectJAutoProxyspring.aop.auto=true#开启CGLIB代理spring.aop.proxy-target-class=true

转载地址:http://jszws.baihongyu.com/

你可能感兴趣的文章
docker-daemon.json各配置详解
查看>>
Mac 下docker路径 /var/lib/docker不存在问题
查看>>
Docker(一)使用阿里云容器镜像服务
查看>>
Docker(二) 基础命令
查看>>
Docker(三) 构建镜像
查看>>
Spring 全家桶注解一览
查看>>
JDK1.8-Stream API使用
查看>>
cant connect to local MySQL server through socket /tmp/mysql.sock (2)
查看>>
vue中的状态管理 vuex store
查看>>
Maven之阿里云镜像仓库配置
查看>>
Maven:mirror和repository 区别
查看>>
微服务网关 Spring Cloud Gateway
查看>>
SpringCloud Feign的使用方式(一)
查看>>
SpringCloud Feign的使用方式(二)
查看>>
关于Vue-cli+ElementUI项目 打包时排除Vue和ElementUI
查看>>
Vue 路由懒加载根据根路由合并chunk块
查看>>
vue中 不更新视图 四种解决方法
查看>>
MySQL 查看执行计划
查看>>
OpenGL ES 3.0(四)图元、VBO、VAO
查看>>
OpenGL ES 3.0(五)纹理
查看>>