Spring中注解方式的異步請求
一、Servlet3.0異步請求
@WebServlet(value = "/async", asyncSupported = true)
public class HelloAsyncServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//1、設(shè)置支持異步處理asyncSupported = true
//2、開啟異步模式
System.out.println("主線程開始:" + Thread.currentThread() + "==>" + System.currentTimeMillis());
AsyncContext startAsync = req.startAsync();
//3、業(yè)務(wù)邏輯進行異步處理,開始異步處理
startAsync.start(new Runnable() {
@Override
public void run() {
try {
System.out.println("副線程開始:" + Thread.currentThread() + "==>" + System.currentTimeMillis());
sayHello();
//獲取到異步的上下文
AsyncContext asyncContext = req.getAsyncContext();
startAsync.complete();
ServletResponse response = asyncContext.getResponse();
response.getWriter().write("hello async!");
System.out.println("副線程結(jié)束:" + Thread.currentThread() + "==>" + System.currentTimeMillis());
} catch (Exception e) {
e.printStackTrace();
}
}
});
System.out.println("主線程結(jié)束:" + Thread.currentThread() + "==>" + System.currentTimeMillis());
}
public void sayHello() throws Exception {
System.out.println(Thread.currentThread() + "processing...");
Thread.sleep(3000);
}
}
打印結(jié)果:

二、SpringMVC的異步請求
返回Callable
@Controller
public class AsyncController {
@ResponseBody
@RequestMapping("/async01")
public Callable<String> async01(){
System.out.println("主線程開始:" + Thread.currentThread() + "==>" + System.currentTimeMillis());
Callable<String> callable = new Callable<String>() {
public String call() throws Exception {
System.out.println("副線程開始:" + Thread.currentThread() + "==>" + System.currentTimeMillis());
Thread.sleep(2000);
System.out.println("副線程開始:" + Thread.currentThread() + "==>" + System.currentTimeMillis());
return "async01";
}
};
System.out.println("主線程結(jié)束:" + Thread.currentThread() + "==>" + System.currentTimeMillis());
return callable;
}
}
- 控制器返回
Callable - Spring進行異步處理,將
Callable提交給TaskExecutor,使用一個隔離的線程進行執(zhí)行 DispatcherServlet和所有的Filter退出Web容器的線程,但是response保持打開狀態(tài)Callable返回結(jié)果,SpreingMVC將請求重新派發(fā)給容器,恢復(fù)之前的處理,Callable返回值就是目標(biāo)方法的返回值- 根據(jù)
Callable返回的結(jié)果,SpringMVC繼續(xù)進行視圖渲染流程等(從收到請求到視圖渲染)
輸出結(jié)果:

返回DeferredResult
模擬一個消息中間件
public class DeferredResultQueue {
private static Queue<DeferredResult<Object>> queue = new ConcurrentLinkedDeque<DeferredResult<Object>>();
public static void save(DeferredResult<Object> deferredResult){
queue.add(deferredResult);
}
public static DeferredResult<Object> get(){
return queue.poll();
}
}
/createOrder請求會暫時保存DeferredResultQueue中,/create請求會獲取DeferredResultQueue中的請求,進行業(yè)務(wù)邏輯的處理并返回結(jié)果
@Controller
public class AsyncController {
@ResponseBody
@RequestMapping("/createOrder")
public DeferredResult<Object> createOrder(){
DeferredResult<Object> deferredResult = new DeferredResult<Object>((long)3000,"create fail");
DeferredResultQueue.save(deferredResult);
return deferredResult;
}
@ResponseBody
@RequestMapping("/create")
public String create(){
String order = UUID.randomUUID().toString();
DeferredResult<Object> deferredResult = DeferredResultQueue.get();
deferredResult.setResult(order);
return "success:" + order;
}
}
到此這篇關(guān)于Spring中注解方式的異步請求的文章就介紹到這了,更多相關(guān)注解方式的異步請求內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
分析Java中ArrayList與LinkedList列表結(jié)構(gòu)的源碼
這篇文章主要介紹了Java中ArrayList與LinkedList列表結(jié)構(gòu)的源碼,文章最后對LinkedList和ArrayList以及Vector的特性有一個對比總結(jié),需要的朋友可以參考下2016-05-05
使用Java實現(xiàn)類似Comet風(fēng)格的web app
這篇文章主要介紹了使用Java實現(xiàn)類似Comet風(fēng)格的web app的方法,包括客戶端的響應(yīng)和XML解析等功能,需要的朋友可以參考下2015-11-11
Springboot實現(xiàn)接口傳輸加解密的步驟詳解
這篇文章主要給大家詳細(xì)介紹了Springboot實現(xiàn)接口傳輸加解密的操作步驟,文中有詳細(xì)的圖文解釋和代碼示例供大家參考,對大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2023-09-09
Java性能優(yōu)化之?dāng)?shù)據(jù)結(jié)構(gòu)實例代碼
這篇文章主要介紹了Java性能優(yōu)化之?dāng)?shù)據(jù)結(jié)構(gòu)實例代碼,具有一定借鑒價值,需要的朋友可以參考下2018-01-01

