java 异步提交 退出,springmvc处理异步请求的示例

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

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

当controller返回值是Callable的时候,springmvc就会启动一个线程将Callable交给TaskExecutor去处理

然后DispatcherServlet还有所有的spring拦截器都退出主线程,然后把response保持打开的状态

当Callable执行结束之后,springmvc就会重新启动分配一个request请求,然后DispatcherServlet就重新调用和处理Callable异步执行的返回结果,然后返回视图

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

springmvc配置异步请求

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

xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

http://java.sun.com/xml/ns/javaee

http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"

version="3.0">

...

2.在web.xml的servlet还有filter添加true子节点

dispatcher

org.springframework.web.servlet.DispatcherServlet

contextConfigLocation

classpath*:META-INF/dispatcher-context.xml

1

true

CharacterEncodingFilter

org.springframework.web.filter.CharacterEncodingFilter

true

encoding

UTF-8

forceEncoding

true

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

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

@RequestMapping("/mvc25")

public Callable mvc25() {

return new Callable() {

@Override

public String call() throws Exception {

Thread.sleep(2000);

return "task/task";

}

};

}

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

@RequestMapping("/mvc26")

@ResponseBody

public Callable mvc26() {

return new Callable() {

@Override

public String call() throws Exception {

Thread.sleep(2000);

return "hello task";

}

};

}

可以自定义客户端超时间

@RequestMapping("/mvc27")

@ResponseBody

public WebAsyncTask mvc27() {

Callable callable = new Callable() {

@Override

public String call() throws Exception {

Thread.sleep(10000);

return "hello task";

}

};

return new WebAsyncTask(10000, callable);

}

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

@RequestMapping("/mvc28")

@ResponseBody

public Callable mvc28() {

Callable callable = new Callable() {

@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的作用是返回一个实例给其他线程来处理这个异步请求。

@RequestMapping("/mvc29")

@ResponseBody

public DeferredResult mvc29() {

DeferredResult deferredResult = new DeferredResult();

dealInOtherThread(deferredResult);

return deferredResult;

}

private void dealInOtherThread(DeferredResult 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可以注册一个请求完成的回调函数。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值