springmvc處理異步請求的示例
springmvc 3.2開始就支持servlet3.0的異步請求。平常我們請求一個(gè)controller一般都是同步的,如果在代碼執(zhí)行中,遇到耗時(shí)的業(yè)務(wù)操作,那servlet容器線程就會被鎖死,當(dāng)有其他請求進(jìn)來的時(shí)候就會受堵了。
springmvc3.2之后支持異步請求,能夠在controller中返回一個(gè)Callable或者DeferredResult。當(dāng)返回Callable的時(shí)候,大概的執(zhí)行過程如下:
- 當(dāng)controller返回值是Callable的時(shí)候,springmvc就會啟動一個(gè)線程將Callable交給TaskExecutor去處理
- 然后DispatcherServlet還有所有的spring攔截器都退出主線程,然后把response保持打開的狀態(tài)
- 當(dāng)Callable執(zhí)行結(jié)束之后,springmvc就會重新啟動分配一個(gè)request請求,然后DispatcherServlet就重新調(diào)用和處理Callable異步執(zhí)行的返回結(jié)果,然后返回視圖
DeferredResult的執(zhí)行過程和Callable差不多,唯一不同的時(shí)候,DeferredResult是由應(yīng)用程序其他線程執(zhí)行返回結(jié)果,而Callable是由TaskExecutor執(zhí)行返回結(jié)果。
springmvc配置異步請求
1.需要在web.xml加上servlet3.0的scheme庫
<web-app xmlns="http://java.sun.com/xml/ns/javaee" 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"> ... </web-app>
2.在web.xml的servlet還有filter添加<asyncsupported>true</async-supported>子節(jié)點(diǎn)
<!-- 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中執(zhí)行異步請求了
利用Callable執(zhí)行異步請求,并返回視圖
@RequestMapping("/mvc25")
public Callable<String> mvc25() {
return new Callable<String>() {
@Override
public String call() throws Exception {
Thread.sleep(2000);
return "task/task";
}
};
}
利用Callable執(zhí)行異步請求,并把請求結(jié)果通過@response由httpmessageconverter進(jìn)行轉(zhuǎn)化返回客戶端
@RequestMapping("/mvc26")
@ResponseBody
public Callable<String> mvc26() {
return new Callable<String>() {
@Override
public String call() throws Exception {
Thread.sleep(2000);
return "hello task";
}
};
}
可以自定義客戶端超時(shí)間
@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);
}
如果在線程的執(zhí)行過程中,遇到異常,處理過程和普通請求的一樣,你可以用@ExceptionHandler來處理或者定義全局的HandlerExceptionResolver來處理
@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的作用是返回一個(gè)實(shí)例給其他線程來處理這個(gè)異步請求。
@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的時(shí)候就會觸發(fā)springmvc分配一個(gè)request到DispatcherServlet,然后DispatcherServlet處理DeferredResult的返回結(jié)果,并返回視圖。
DeferredResult還提供了其他返回來處理線程請求,例如onTimeout(Runnable) 還有onCompletion(Runnable),onTimeout可以注冊一個(gè)線程回調(diào),當(dāng)請求延時(shí)的時(shí)候的回調(diào)函數(shù),onCompletion可以注冊一個(gè)請求完成的回調(diào)函數(shù)。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
springBoot 過濾器去除請求參數(shù)前后空格實(shí)例詳解
這篇文章主要為大家介紹了springBoot 過濾器去除請求參數(shù)前后空格實(shí)例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11
Java?HashMap詳解及實(shí)現(xiàn)原理
Java?HashMap是Java集合框架中最常用的實(shí)現(xiàn)Map接口的數(shù)據(jù)結(jié)構(gòu),它使用哈希表實(shí)現(xiàn),允許null作為鍵和值,可以存儲不同類型的鍵值對,在Java中,HashMap被廣泛應(yīng)用于各種場景,如緩存、數(shù)據(jù)庫連接池、路由器等,文中有詳細(xì)的代碼示例,需要的朋友可以參考下2023-05-05
SpringSecurity動態(tài)加載用戶角色權(quán)限實(shí)現(xiàn)登錄及鑒權(quán)功能
這篇文章主要介紹了SpringSecurity動態(tài)加載用戶角色權(quán)限實(shí)現(xiàn)登錄及鑒權(quán)功能,很多朋友感覺這個(gè)功能很難,今天小編通過實(shí)例代碼給大家講解,需要的朋友可以參考下2019-11-11
Spring Cloud EureKa Ribbon 服務(wù)注冊發(fā)現(xiàn)與調(diào)用
這篇文章主要介紹了Spring Cloud EureKa Ribbon 服務(wù)注冊發(fā)現(xiàn)與調(diào)用,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-02-02
springboot項(xiàng)目開啟https協(xié)議的項(xiàng)目實(shí)現(xiàn)
本文主要介紹了springboot項(xiàng)目開啟https協(xié)議的項(xiàng)目實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-07-07
Java后端限制頻繁請求和重復(fù)提交的實(shí)現(xiàn)
很多用戶會請求過于頻繁或者是多次重復(fù)提交數(shù)據(jù),本文主要介紹了Java后端限制頻繁請求和重復(fù)提交的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-04-04
SpringMVC配置javaConfig及StringHttpMessageConverter示例
這篇文章主要介紹了SpringMVC配置javaConfig及StringHttpMessageConverter實(shí)現(xiàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07

