我想,可能不止我一个人误解于Spring Boot 中@EnableAsync/@Async
异步执行方法的注解。
错误以为它们能够像@EnableScheduling/@Scheduled
注解一样,在方法标注了@Async
后,能够自动运行,但且只会被执行一次!
但实际上并不是这样的 :(
对于在Spring Boot应用中,框架对于标注了@EnableAsync/@Async
注解的方法,一般介绍是通过proxy
代理类和AOP
切面技术进行实现,所以,需要主动的跨proxy/Bean间调用,这样的方法调用路径,才会被AOP
触发,进而包裹被调用方法去异步执行。
所以,类似其他网上博文所说的,不能在本Bean中自己调用自己的异步方法,或在本Bean @PostConstruct
标注方法中调用自身标注了@Async
的方法,都不能够异步运行。
这是因为异步方法调用,没有通过“proxy”类间调用,不能够正确被AOP机制截获
扩展
在Spring Boot中,如果只想让方法启动时执行一次,或调用时异步执行,那么或在业务流中在Bean间方法调用触发(注意Bean装配依赖),或在Spring Boot启动ApplicationListener<ContextRefreshedEvent>/ApplicationRunner/CommandLineRunner
中进行定制方法。