时间:2023-02-26 08:36:09 | 栏目:JAVA代码 | 点击:次
Spring MVC 3.2开始引入了基于Servlet 3的异步请求处理。相比以前,控制器方法已经不一定需要返回一个值,而是可以返回一个java.util.concurrent.Callable的对象,并通过Spring MVC所管理的线程来产生返回值。与此同时,Servlet容器的主线程则可以退出并释放其资源了,同时也允许容器去处理其他的请求。通过一个TaskExecutor,Spring MVC可以在另外的线程中调用Callable。当Callable返回时,请求再携带Callable返回的值,再次被分配到Servlet容器中恢复处理流程。以下代码给出了一个这样的控制器方法作为例子:
@RequestMapping(method=RequestMethod.POST) public CallableprocessUpload(final MultipartFile file) { return new Callable() { public String call() throws Exception { // ... return "someView"; } }; }
另一个选择,是让控制器方法返回一个DeferredResult的实例。这种场景下,返回值可以由任何一个线程产生,也包括那些不是由Spring MVC管理的线程。举个例子,返回值可能是为了响应某些外部事件所产生的,比如一条JMS的消息,一个计划任务,等等。以下代码给出了一个这样的控制器作为例子:
@RequestMapping("/quotes") @ResponseBody public DeferredResultquotes() { DeferredResultdeferredResult = new DeferredResult(); // Save the deferredResult somewhere.. return deferredResult; } // In some other thread... deferredResult.setResult(data);
如果对Servlet 3.0的异步请求处理特性没有了解,理解这个特性可能会有点困难。因此,阅读一下前者的文档将会很有帮助。
以下给出了这个机制运作背后的一些原理:
一个servlet请求ServletRequest可以通过调用request.startAsync()方法而进入异步模式。这样做的主要结果就是该servlet以及所有的过滤器都可以结束,但其响应(response)会留待异步处理结束后再返回调用request.startAsync()方法会返回一个AsyncContext对象,可用它对异步处理进行进一步的控制和操作。比如说它也提供了一个与转向(forward)很相似的dispatch方法,只不过它允许应用恢复Servlet容器的请求处理进程ServletRequest提供了获取当前DispatherType的方式,后者可以用来区别当前处理的是原始请求、异步分发请求、转向,或是其他类型的请求分发类型。
有了上面的知识,下面可以来看一下Callable的异步请求被处理时所依次发生的事件:
这里暂抛开某些场景webSocket的解决方案。
举一个生活中的列子来说明长轮询比轮询好在哪里:电商云集的时代,大家肯定都有查询快递的经历,怎么最快知道快递的进度呢?polling和long polling的方式分别如下:
如果在发散的触类旁通一下,long polling的方式和发布订阅的模式有点类似之处,只是每次拿到了发布的结果之后需要再次发起消息订阅
因为DeferredResult技术,所以使得long polling不会一直占用容器资源,使得长轮询成为可能。长轮询的应用有很多,简述下就是:需要及时知道某些消息的变更的场景都可以用长轮询来解决,当然,你可能又想起了发布订阅了,哈哈
apollo的具体做法可见
多个请求的结果,使用另一个请求控制他的响应返回。本实例构建在spring boot 1.5.7上。
/** * Created by kl on 2017/9/27. * Content : */ @RestController @RequestMapping("/async") public class AsyncController { final Map deferredResultMap=new ConcurrentReferenceHashMap<>(); @GetMapping("/longPolling") public DeferredResultlongPolling(){ DeferredResultdeferredResult=new DeferredResult(0L); deferredResultMap.put(deferredResult.hashCode(),deferredResult); deferredResult.onCompletion(()->{ deferredResultMap.remove(deferredResult.hashCode()); System.err.println("还剩"+deferredResultMap.size()+"个deferredResult未响应"); }); return deferredResult; } @GetMapping("/returnLongPollingValue") public void returnLongPollingValue(){ for (Map.Entry entry:deferredResultMap.entrySet()){ entry.getValue().setResult("kl"); } } }
/** * Created by kl on 2017/9/27. * Content : */ @FeignClient(url = "localhost:8976",name = "async") public interface AsyncFeginService { @GetMapping("/async/longPolling") String longPolling(); @GetMapping("/async/returnLongPollingValue") void returnLongPollingValue(); }
@RunWith(SpringRunner.class) @SpringBootTest public class LongPollingdemoApplicationTests { @Autowired AsyncFeginService asyncFeginService; /** * 模拟多个浏览器客户端发起长轮询请求,等待testLongPolling测试用例请求通知服务端返回各浏览器的请求结果 * @throws Exception */ @Test public void contextLoads() throws Exception{ ExecutorService executorService=Executors.newFixedThreadPool(4); for (int i=0;i<=3;i++){ executorService.execute(()->{ String kl=asyncFeginService.longPolling(); System.err.println("收到响应:"+kl); }); } System.in.read(); } /** * 通知服务端返回上个测试的长轮询结果 */ @Test public void testLongPolling(){ asyncFeginService.returnLongPollingValue(); } }
测试时,先启动contextLoads会发起四个异步请求,一直等待请求结果响应,直到testLongPolling通知服务端返回deferredResult的值。