官方文档中说DeferredResult和Callable都是为了异步生成返回值提供基本的支持。简单来说就是一个请求进来,如果你使用了DeferredResult或者Callable,在没有得到返回数据之前,DispatcherServlet和所有Filter就会退出Servlet容器线程,但响应保持打开状态,一旦返回数据有了,这个DispatcherServlet就会被再次调用并且处理,以异步产生的方式,向请求端返回值。
这么做的好处就是请求不会长时间占用服务连接池,提高服务器的吞吐量。
Callable
Callable的实现比较简单,call()方法的返回值就是服务端返回给请求端的数据。
package com.imooc.demo.web.async;
import com.imooc.demo.web.async.two.DeferredResultHolder;
import com.imooc.demo.web.async.two.MockQueue;
import org.apache.commons.lang.RandomStringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.context.request.async.DeferredResult;
import java.util.concurrent.Callable;
/**
* 异步处理rest服务
*/
@Controller
@RequestMapping("/async")
public class AsyncController {
Logger logger = LoggerFactory.getLogger(AsyncController.class);
/**
* 使用Runnable异步处理rest服务
* 弊端:
* 1. 比如消息中间件. 请求 与 响应的线程并不是同一个线程.
* 2. 可以使用deferredResult对象来解决
* @return
*/
@RequestMapping("")
public Callable<String> asyncHandlerMethod() {
logger.info("主线程执行");
Callable<String> result = new Callable<String>() {
@Override
public String call() throws Exception {
logger.info("副线程执行");
Thread.sleep(1000);
logger.info("副线程返回");
return null;
}
};