SpringBoot使用@Async注解失效分析

有时候在使用的过程中@Async注解会失效(原因和@Transactional注解有时候会失效的原因一样)。

下面定义一个Service:

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


定义Controller:


执行方法,返回结果:


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

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

网上有不少博客说解决方法就是将要异步执行的方法单独抽取成一个类,我个人感觉这个不太符合实际项目开发需求。

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

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

工具类可以参考我的另一篇博客:https://blog.csdn.net/Dongguabai/article/details/80788646


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


解决方式二:开启cglib代理,手动获取Spring代理类

在启动类上加上:


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

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


运行结果:


OK,问题完美解决!

猜你喜欢

转载自blog.csdn.net/dongguabai/article/details/80788585