控制器中的弹簧启动@Async方法同步执行
我的[基本]Spring Boot应用程序接受来自浏览器的请求,通过发送并应立即收到响应 - 例如“您的请求已排队”。为了实现这一点,我编写了一个控制器:jQuery.get()
@Controller
public class DoSomeWorkController {
@Autowired
private final DoWorkService workService;
@RequestMapping("/doSomeWork")
@ResponseBody
public String doSomeWork() {
workService.doWork(); // time consuming operation
return "Your request has been queued.";
}
}
该类实现了一个接口,非常简单。它具有执行耗时任务的单一方法。我不需要从此服务呼叫中返回任何内容,因为对于失败或成功的情况,将在工作结束时发送一封电子邮件。因此,它实际上看起来像这样:DoWorkServiceImpl
DoWorkService
@Service
public class DoWorkServiceImpl implements DoWorkService {
@Async("workExecutor")
@Override
public void doWork() {
try {
Thread.sleep(10 * 1000);
System.out.println("completed work, sent email");
}
catch (InterruptedException ie) {
System.err.println(ie.getMessage());
}
}
}
我以为这可以工作,但浏览器的Ajax请求在返回响应之前等待了10秒钟。因此,控制器映射方法似乎正在调用同步注释的内部方法。在传统的Spring应用程序中,我通常将其添加到XML配置中:@Async
<task:annotation-driven />
<task:executor id="workExecutor" pool-size="1" queue-capacity="0" rejection-policy="DISCARD" />
所以我认为在主应用程序类中编写等效项会有所帮助:
@SpringBootApplication
@EnableAsync
public class Application {
@Value("${pool.size:1}")
private int poolSize;;
@Value("${queue.capacity:0}")
private int queueCapacity;
@Bean(name="workExecutor")
public TaskExecutor taskExecutor() {
ThreadPoolTaskExecutor taskExecutor = new ThreadPoolTaskExecutor();
taskExecutor.setMaxPoolSize(poolSize);
taskExecutor.setQueueCapacity(queueCapacity);
taskExecutor.afterPropertiesSet();
return taskExecutor;
}
public static void main(String[] args) {
SpringApplication.run(Application.class, args);
}
}
这并没有改变行为。Ajax 响应在发送请求 10 秒后仍会到达。我错过了什么?
Spring Boot应用程序可以在这里下载。安装 Maven 后,可以使用简单的命令运行项目:
mvn clean spring-boot:run
注意由于下面@Dave Syer提供的答案,这个问题得到了解决,他指出我的应用程序中缺少我,即使我在上面的代码片段中有一行。@EnableAsync