SpringMVC之-Callable和DeferredResult异步请求

爱被打了一巴掌 2022-04-16 01:38 375阅读 0赞

Callable和DeferredResult均是SpringMVC中处理异步请求的方法;

Callable异步请求步骤:

* 1、控制器返回Callable
* 2、Spring异步处理,将Callable 提交到 TaskExecutor 使用一个隔离的线程进行执行
* 3、DispatcherServlet和所有的Filter退出web容器的线程,但是response 保持打开状态;
* 4、Callable返回结果,SpringMVC将请求重新派发给容器,恢复之前的处理;
* 5、根据Callable返回的结果。SpringMVC继续进行视图渲染流程等(从收请求-视图渲染)。
*
* preHandle…/springmvc-annotation/async01
主线程开始…Thread[http-bio-8081-exec-3,5,main]==>1513932494700
主线程结束…Thread[http-bio-8081-exec-3,5,main]==>1513932494700
=========DispatcherServlet及所有的Filter退出线程============================

  1. ================等待Callable执行==========
  2. 副线程开始...Thread\[MvcAsync1,5,main\]==>1513932494707
  3. 副线程开始...Thread\[MvcAsync1,5,main\]==>1513932496708
  4. ================Callable执行完成==========
  5. ================再次收到之前重发过来的请求========
  6. preHandle.../springmvc-annotation/async01
  7. postHandle...(Callable的之前的返回值就是目标方法的返回值)
  8. afterCompletion...

异步的拦截器:
1)、原生API的AsyncListener
2)、SpringMVC:实现AsyncHandlerInterceptor;
* @return


一、返回Callable

新建AsyncController .java

  1. package com.atguigu.controller;
  2. import java.util.UUID;
  3. import java.util.concurrent.Callable;
  4. import org.springframework.stereotype.Controller;
  5. import org.springframework.web.bind.annotation.RequestMapping;
  6. import org.springframework.web.bind.annotation.ResponseBody;
  7. import org.springframework.web.context.request.async.DeferredResult;
  8. import com.atguigu.service.DeferredResultQueue;
  9. @Controller
  10. public class AsyncController {
  11. @ResponseBody
  12. @RequestMapping("/async01")
  13. public Callable<String> async01(){
  14. System.out.println("主线程开始..."+Thread.currentThread()+"==>"+System.currentTimeMillis());
  15. Callable<String> callable = new Callable<String>() {
  16. @Override
  17. public String call() throws Exception {
  18. System.out.println("副线程开始..."+Thread.currentThread()+"==>"+System.currentTimeMillis());
  19. Thread.sleep(2000);
  20. System.out.println("副线程结束..."+Thread.currentThread()+"==>"+System.currentTimeMillis());
  21. return "Callable<String> async01()";
  22. }
  23. };
  24. System.out.println("主线程结束..."+Thread.currentThread()+"==>"+System.currentTimeMillis());
  25. return callable;
  26. }
  27. }

运行tomcat结果:

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2ppbmhhaWppbmc_size_16_color_FFFFFF_t_70

二、返回 DeferredResult

1、新建一个队列

  1. package com.atguigu.service;
  2. import java.util.Queue;
  3. import java.util.concurrent.ConcurrentLinkedQueue;
  4. import org.springframework.web.context.request.async.DeferredResult;
  5. public class DeferredResultQueue {
  6. private static Queue<DeferredResult<Object>> queue = new ConcurrentLinkedQueue<DeferredResult<Object>>();
  7. public static void save(DeferredResult<Object> deferredResult){
  8. queue.add(deferredResult);
  9. }
  10. public static DeferredResult<Object> get( ){
  11. return queue.poll();
  12. }
  13. }

2、新建AsyncController.java

  1. package com.atguigu.controller;
  2. import java.util.UUID;
  3. import java.util.concurrent.Callable;
  4. import org.springframework.stereotype.Controller;
  5. import org.springframework.web.bind.annotation.RequestMapping;
  6. import org.springframework.web.bind.annotation.ResponseBody;
  7. import org.springframework.web.context.request.async.DeferredResult;
  8. import com.atguigu.service.DeferredResultQueue;
  9. @Controller
  10. public class AsyncController {
  11. @ResponseBody
  12. @RequestMapping("/createOrder")
  13. public DeferredResult<Object> createOrder(){
  14. DeferredResult<Object> deferredResult = new DeferredResult<>((long)3000, "create fail...");
  15. DeferredResultQueue.save(deferredResult);
  16. return deferredResult;
  17. }
  18. @ResponseBody
  19. @RequestMapping("/create")
  20. public String create(){
  21. //创建订单
  22. String order = UUID.randomUUID().toString();
  23. DeferredResult<Object> deferredResult = DeferredResultQueue.get();
  24. deferredResult.setResult(order);
  25. return "success===>"+order;
  26. }
  27. }

运行tomcat结果:

查看订单号:

20181115110906737.png

20181115110938737.png

结果说明:说明两个方法返回的订单号相同,说明如果有另一个线程给DeferredResult赋值后,DeferredResult在感知到自己的对象被赋值后就返回页面成功;

======以下于你或许是个好消息======

好消息就是:欢迎访问下面的博客网站哈哈哈……

网站名称:Java学习笔记网 (点击进入)

url:https://www.javaxxbj.com/ (点击进入)

网站特点:

  1. java主要网站的导航目录
  2. 你可以记录自己的博客,并可以控制显示和隐藏,可利于管理啦!!!
  3. 可以添加收藏各个网站的链接!!!
  4. 甚至也可以文章收藏,点赞,关注,查看我的消息等功能哦!!1

看一小点点的截图:

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2ppbmhhaWppbmc_size_16_color_FFFFFF_t_70 1

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2ppbmhhaWppbmc_size_16_color_FFFFFF_t_70 2

或可一试哦!

发表评论

表情:
评论列表 (有 0 条评论,375人围观)

还没有评论,来说两句吧...

相关阅读

    相关 使用DeferredResult异步处理请求

         我们在实际开发的过程中,可能会遇到需要长轮循的情况。      长轮询指的是客户端向服务端发送一个请求,然后服务端将该请求hold住,另起线程进行处理,处理完后,再