32

我正在使用 Spring 4 并且我注意到一个奇怪的行为......如果我从普通实例方法多次调用异步方法,那么它们都在不同的线程中调用并在随机时间完成。但是,如果我从另一个异步方法多次调用一个异步方法,那么它们会按顺序完成。我有这样的事情:

@Async
public void nonAsyncMethod() {
  for (int i = 0; i < 30; i++) {
     asyncMethod();
  }
}

@Async
public void asyncMethod() {
   ... something here
}

我正在使用默认的异步执行器。我应该使用不同的吗?但是,这个执行器不重用任何线程,并且每次都启动另一个线程,所以应该没问题......这可能只是巧合吗?但是我已经尝试了超过 10 次,如果我将第一种方法恢复为非异步,那么它们会随机完成

4

1 回答 1

73

您所描述的是 Spring AOP 的一个经典陷阱。

简而言之,为了让 Spring 能够提供异步行为,它需要在运行时为您的类创建代理。然后,代理在调用您的代码之前和/或之后执行它需要做的任何事情。但是在您的情况下,代理机制并未应用于第二种方法。

当您的类的 bean 通过 Spring 注入到其他组件时,Spring 确实注入了代理。因此调用了代理的相关方法。但是,当您从类内部调用方法时,Spring AOP 的限制意味着代理永远不会发挥作用,而是调用常规方法 - 没有额外的功能。

这就是为什么asyncMethod总是与调用它的同一个类中的另一个方法在同一个线程上执行。

查看这篇优秀的博客文章以及Spring 文档的这一部分。

有一些方法可以解决这个问题(看看这个),不需要你重构你的代码,但是如果你想让 async 无论如何都可以在这两种方法上工作,最简单的方法是将第二种方法重构到另一个类中.

于 2014-07-23T05:20:54.160 回答