java - 从另一个异步方法调用的 Spring 异步方法

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

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

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

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

最佳答案

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

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

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

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

查看 this优秀的博文以及this Spring 文档的一部分。

有一些解决问题的方法(查看 this)不需要你重构你的代码,但是如果你想让 async 无论如何都在这两种方法上工作,最简单的事情就是重构第二种方法进入另一个类。

https://stackoverflow.com/questions/24898547/

相关文章:

java - 为什么 Spring 的 jdbcTemplate.batchUpdate() 这么慢

java - 我无法使用断点进行调试

java - 为什么来自 Spring 的 BCryptPasswordEncoder 为相同的输入

java - AspectJ 表达式在切入点错误中给出正式的未绑定(bind)

java - 如何在spring mvc中使用带有freemarker的消息?

java - 无法将 boolean 值设置为 null

java - 比较 SpEL 中的枚举

java - @ControllerAdvice 在处理异常方面比 @ExceptionHandle

java - 在 Spring Boot 中读取环境变量

spring - 配置多个数据源后无法设置JPA命名策略(Spring 1.4.1/Hibernat