springmvc 处理异步请求

秒速五厘米 2022-07-30 16:23 313阅读 0赞

springmvc 3.2开始就支持servlet3.0的异步请求。平常我们请求一个controller一般都是同步的,如果在代码执行中,遇到耗时的业务操作,那servlet容器线程就会被锁死,当有其他请求进来的时候就会受堵了。

springmvc3.2之后支持异步请求,能够在controller中返回一个Callable或者DeferredResult。当返回Callable的时候,大概的执行过程如下:

  1. 当controller返回值是Callable的时候,springmvc就会启动一个线程将Callable交给TaskExecutor去处理
  2. 然后DispatcherServlet还有所有的spring拦截器都退出主线程,然后把response保持打开的状态
  3. 当Callable执行结束之后,springmvc就会重新启动分配一个request请求,然后DispatcherServlet就重新调用和处理Callable异步执行的返回结果,然后返回视图

DeferredResult的执行过程和Callable差不多,唯一不同的时候,DeferredResult是由应用程序其他线程执行返回结果,而Callable是由TaskExecutor执行返回结果。

springmvc配置异步请求

1.需要在web.xml加上servlet3.0的scheme库

  1. <web-app xmlns="http://java.sun.com/xml/ns/javaee"
  2. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  3. http://java.sun.com/xml/ns/javaee
  4. http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
  5. version="3.0">
  6. ...
  7. </web-app>

2.在web.xml的servlet还有filter添加<asyncsupported>true</async-supported>子节点

  1. <!-- springMVC的Servlet配置 -->
  2. <servlet>
  3. <servlet-name>dispatcher</servlet-name>
  4. <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
  5. <init-param>
  6. <param-name>contextConfigLocation</param-name>
  7. <param-value>classpath*:META-INF/dispatcher-context.xml</param-value>
  8. </init-param>
  9. <load-on-startup>1</load-on-startup>
  10. <async-supported>true</async-supported>
  11. </servlet>
  12. <!-- 编码拦截 -->
  13. <filter>
  14. <filter-name>CharacterEncodingFilter</filter-name>
  15. <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
  16. <async-supported>true</async-supported>
  17. <init-param>
  18. <param-name>encoding</param-name>
  19. <param-value>UTF-8</param-value>
  20. </init-param>
  21. <init-param>
  22. <param-name>forceEncoding</param-name>
  23. <param-value>true</param-value>
  24. </init-param>
  25. </filter>

3.然后就可以在controller中执行异步请求了

利用Callable执行异步请求,并返回视图

  1. @RequestMapping("/mvc25")
  2. public Callable<String> mvc25() {
  3. return new Callable<String>() {
  4. @Override
  5. public String call() throws Exception {
  6. Thread.sleep(2000);
  7. return "task/task";
  8. }
  9. };
  10. }

利用Callable执行异步请求,并把请求结果通过@response由httpmessageconverter进行转化返回客户端

  1. @RequestMapping("/mvc26")
  2. @ResponseBody
  3. public Callable<String> mvc26() {
  4. return new Callable<String>() {
  5. @Override
  6. public String call() throws Exception {
  7. Thread.sleep(2000);
  8. return "hello task";
  9. }
  10. };
  11. }

可以自定义客户端超时间

  1. @RequestMapping("/mvc27")
  2. @ResponseBody
  3. public WebAsyncTask<String> mvc27() {
  4. Callable<String> callable = new Callable<String>() {
  5. @Override
  6. public String call() throws Exception {
  7. Thread.sleep(10000);
  8. return "hello task";
  9. }
  10. };
  11. return new WebAsyncTask<String>(10000, callable);
  12. }

如果在线程的执行过程中,遇到异常,处理过程和普通请求的一样,你可以用@ExceptionHandler来处理或者定义全局的HandlerExceptionResolver来处理

  1. @RequestMapping("/mvc28")
  2. @ResponseBody
  3. public Callable<String> mvc28() {
  4. Callable<String> callable = new Callable<String>() {
  5. @Override
  6. public String call() throws Exception {
  7. Thread.sleep(2000);
  8. throw new RuntimeException();
  9. }
  10. };
  11. return callable;
  12. }
  13. @ExceptionHandler(RuntimeException.class)
  14. @ResponseBody
  15. public JSONObject handlerException(){
  16. JSONObject jsonObject = new JSONObject();
  17. jsonObject.put("aaa", 123);
  18. return jsonObject ;
  19. }

还可以通过返回DeferredResult返回,DeferredResult的作用是返回一个实例给其他线程来处理这个异步请求。

  1. @RequestMapping("/mvc29")
  2. @ResponseBody
  3. public DeferredResult<String> mvc29() {
  4. DeferredResult<String> deferredResult = new DeferredResult<String>();
  5. dealInOtherThread(deferredResult);
  6. return deferredResult;
  7. }
  8. private void dealInOtherThread(DeferredResult<String> deferredResult) {
  9. try {
  10. Thread.sleep(2000);
  11. } catch (InterruptedException e) {
  12. // TODO Auto-generated catch block
  13. e.printStackTrace();
  14. }
  15. deferredResult.setResult("hello task");
  16. }

dealInOtherThread处理完成,setResult的时候就会触发springmvc分配一个request到DispatcherServlet,然后DispatcherServlet处理DeferredResult的返回结果,并返回视图。

DeferredResult还提供了其他返回来处理线程请求,例如onTimeout(Runnable) 还有onCompletion(Runnable),onTimeout可以注册一个线程回调,当请求延时的时候的回调函数,onCompletion可以注册一个请求完成的回调函数。

发表评论

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

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

相关阅读

    相关 springmvc异步处理请求

    有两种情况,第一种是业务逻辑复杂,但不需要业务逻辑的结果,第二种是需要返回业务逻辑的处理结果 第一种比较简单,利用多线程处理业务逻辑,或者利用spring中@Asyn注