我正在使用
Spring 4,我注意到一个奇怪的行为…如果我从一个常规实例方法多次调用异步方法,那么它们都被调用在不同的线程中,并在随机时间完成.但是如果我从另一个异步方法多次调用异步方法,那么它们按顺序完成.我有这样的东西:
@Async public void nonAsyncMethod() { for (int i = 0; i < 30; i++) { asyncMethod(); } } @Async public void asyncMethod() { ... something here }
我正在使用默认的异步执行器.我应该用不同的吗?然而,这个执行者不要重复使用任何线程,每次启动另一个线程,所以它应该是好的…它只是一个巧合?但是我已经尝试了十多次,如果我恢复到第一种方法的非同步,那么它们随机完成
解决方法
你所描述的是Spring AOP的经典陷阱.
简而言之,对于Spring,可以提供异步行为,它需要在运行时为您的类创建一个代理.代理然后在调用您的代码之前和/或之后执行任何需要执行的操作.但是在你的情况下,代理机制并没有被应用于第二种方法.
当你的类的bean通过Spring注入到其他组件中时,Spring真的会注入代理.因此调用代理的相关方法.但是,当您从类中调用方法时,Spring AOP的局限性意味着代理不会发挥作用,而是调用常规方法,而无需额外的功能.
这就是为什么asyncMethod总是在与调用它的同一个类中的另一个方法相同的线程上执行.
查看this优秀博客文章以及this部分的Spring文档.
有一些方法可以解决不需要重构代码的问题(参见this),但是如果您希望异步工作在两种方法上,无论什么,最简单的做法是将第二种方法重构为另一类.