springmvc 3.2开始就支持servlet3.0的异步请求。日常咱们请求一个controller通常都是同步的,若是在代码执行中,遇到耗时的业务操做,那servlet容器线程就会被锁死,当有其余请求进来的时候就会受堵了。java
springmvc3.2以后支持异步请求,可以在controller中返回一个Callable或者DeferredResult。当返回Callable的时候,大概的执行过程以下:web
- 当controller返回值是Callable的时候,springmvc就会启动一个线程将Callable交给TaskExecutor去处理
- 而后DispatcherServlet还有全部的spring拦截器都退出主线程,而后把response保持打开的状态
- 当Callable执行结束以后,springmvc就会从新启动分配一个request请求,而后DispatcherServlet就从新调用和处理Callable异步执行的返回结果,而后返回视图
DeferredResult的执行过程和Callable差很少,惟一不一样的时候,DeferredResult是由应用程序其余线程执行返回结果,而Callable是由TaskExecutor执行返回结果。spring
springmvc配置异步请求json
1.须要在web.xml加上servlet3.0的scheme库mvc
2.在web.xml的servlet还有filter添加<asyncsupported>true</async-supported>子节点
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 |
<!-- springMVC的Servlet配置 --> < servlet > < servlet-name >dispatcher</ servlet-name > < servlet-class >org.springframework.web.servlet.DispatcherServlet</ servlet-class > < init-param > < param-name >contextConfigLocation</ param-name > < param-value >classpath*:META-INF/dispatcher-context.xml</ param-value > </ init-param > < load-on-startup >1</ load-on-startup > < async-supported >true</ async-supported > </ servlet > <!-- 编码拦截 --> < filter > < filter-name >CharacterEncodingFilter</ filter-name > < filter-class >org.springframework.web.filter.CharacterEncodingFilter</ filter-class > < async-supported >true</ async-supported > < init-param > < param-name >encoding</ param-name > < param-value >UTF-8</ param-value > </ init-param > < init-param > < param-name >forceEncoding</ param-name > < param-value >true</ param-value > </ init-param > </ filter > |
3.而后就能够在controller中执行异步请求了
利用Callable执行异步请求,并返回视图
1 2 3 4 5 6 7 8 9 10 11 12 |
@RequestMapping ( "/mvc25" ) public Callable<String> mvc25() { return new Callable<String>() { @Override public String call() throws Exception { Thread.sleep( 2000 ); return "task/task" ; } }; } |
利用Callable执行异步请求,并把请求结果经过@response由httpmessageconverter进行转化返回客户端
1 2 3 4 5 6 7 8 9 10 11 12 13 |
@RequestMapping ( "/mvc26" ) @ResponseBody public Callable<String> mvc26() { return new Callable<String>() { @Override public String call() throws Exception { Thread.sleep( 2000 ); return "hello task" ; } }; } |
能够自定义客户端超时间
1 2 3 4 5 6 7 8 9 10 11 12 13 14 |
@RequestMapping ( "/mvc27" ) @ResponseBody public WebAsyncTask<String> mvc27() { Callable<String> callable = new Callable<String>() { @Override public String call() throws Exception { Thread.sleep( 10000 ); return "hello task" ; } }; return new WebAsyncTask<String>( 10000 , callable); } |
若是在线程的执行过程当中,遇到异常,处理过程和普通请求的同样,你能够用@ExceptionHandler来处理或者定义全局的HandlerExceptionResolver来处理
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 |
@RequestMapping ( "/mvc28" ) @ResponseBody public Callable<String> mvc28() { Callable<String> callable = new Callable<String>() { @Override public String call() throws Exception { Thread.sleep( 2000 ); throw new RuntimeException(); } }; return callable; } @ExceptionHandler (RuntimeException. class ) @ResponseBody public JSONObject handlerException(){ JSONObject jsonObject = new JSONObject(); jsonObject.put( "aaa" , 123 ); return jsonObject ; } |
还能够经过返回DeferredResult返回,DeferredResult的做用是返回一个实例给其余线程来处理这个异步请求。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
@RequestMapping ( "/mvc29" ) @ResponseBody public DeferredResult<String> mvc29() { DeferredResult<String> deferredResult = new DeferredResult<String>(); dealInOtherThread(deferredResult); return deferredResult; } private void dealInOtherThread(DeferredResult<String> deferredResult) { try { Thread.sleep( 2000 ); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } deferredResult.setResult( "hello task" ); } |
dealInOtherThread处理完成,setResult的时候就会触发springmvc分配一个request到DispatcherServlet,而后DispatcherServlet处理DeferredResult的返回结果,并返回视图。
DeferredResult还提供了其余返回来处理线程请求,例如onTimeout(Runnable) 还有onCompletion(Runnable),onTimeout能够注册一个线程回调,当请求延时的时候的回调函数,onCompletion能够注册一个请求完成的回调函数。