servlet異步請求的實現(xiàn)
1、什么是servlet異步請求
Servlet 3.0 之前,一個普通 Servlet 的主要工作流程大致如下:
(1)、Servlet 接收到請求之后,可能需要對請求攜帶的數(shù)據(jù)進行一些預(yù)處理;
(2)、調(diào)用業(yè)務(wù)接口的某些方法,以完成業(yè)務(wù)處理;
(3)、根據(jù)處理的結(jié)果提交響應(yīng),Servlet 線程結(jié)束。
其中第二步處理業(yè)務(wù)邏輯時候很可以碰到比較耗時的任務(wù),此時servlet主線程會阻塞等待完成業(yè)務(wù)處理,對于并發(fā)比較大的請求可能會產(chǎn)生性能瓶頸,則servlet3.0之后再此處做了調(diào)整,引入了異步的概念。
(1)、Servlet 接收到請求之后,可能需要對請求攜帶的數(shù)據(jù)進行一些預(yù)處理;
(2)、調(diào)用業(yè)務(wù)接口的某些方法過程中request.startAsync()請求,獲取一個AsyncContext
(3)、緊接著servlet線程退出(回收到線程池),但是響應(yīng)response對象仍舊保持打開狀態(tài),新增線程會使用AsyncContext處理并響應(yīng)結(jié)果。
(4)、AsyncContext處理完成觸發(fā)某些監(jiān)聽通知結(jié)果
2、Servlet異步請求示例
2.1、示例準(zhǔn)備
本示例采用web.xml配置的形式,模擬場景為:筆者所在的it公司每周的工作內(nèi)容,首先研發(fā)總監(jiān)分配給產(chǎn)品、研發(fā)、測試相關(guān)的任務(wù),布置完任務(wù)就出差(模擬請求響應(yīng)),余下的各個小組進行自己任務(wù)操作(模擬的耗時操作),最終出周報完成任務(wù)(異步任務(wù)處理完成的通知)
git地址:https://github.com/liushangzaibeijing/spsm.git 分支:dev_async
2.2、實現(xiàn)自定義的Servlet
/**
* @ClassName AsyncServlet
* @Desc 自定義異步Servlet處理器
* @Author xieqx
* @Date 2020/12/9 15:38
**/
//通過注解的形式開始異步
@WebServlet(urlPatterns = "*.async",asyncSupported = true)
public class AsyncServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doPost(req,resp);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//開啟異步支持
//異步管理上下文
resp.setCharacterEncoding("GBK");
PrintWriter writer = resp.getWriter();
writer.println("周工作任務(wù)布置開始");
AsyncContext asyncContext = req.startAsync();
asyncContext.start(new WeekTask(asyncContext));
//添加監(jiān)聽器 處理完成監(jiān)聽
asyncContext.addListener(new AsyncListener() {
@Override
public void onComplete(AsyncEvent asyncEvent) throws IOException {
System.out.println("工作在"+new Date()+"處理完成");
}
@Override
public void onTimeout(AsyncEvent asyncEvent) throws IOException {
System.out.println("工作在"+new Date()+"處理超時");
}
@Override
public void onError(AsyncEvent asyncEvent) throws IOException {
System.out.println("工作在"+new Date()+"處理出錯");
}
@Override
public void onStartAsync(AsyncEvent asyncEvent) throws IOException {
System.out.println("工作在"+new Date()+"處理開始");
}
});
writer.println("周工作任務(wù)布置完成");
writer.flush();
}
}開啟異步支持(默認異步支持不開啟)有兩種方式:
使用注解

web.xml配置
<servlet> <servlet-name>asyncServlet</servlet-name> <servlet-class>com.xiu.async.servlet.AsyncServlet</servlet-class> <!-- 開啟servlet的異步請求操作 --> <async-supported>true</async-supported> </servlet> <servlet-mapping> <servlet-name>asyncServlet</servlet-name> <url-pattern>*.async</url-pattern> </servlet-mapping>
上述代碼中通過request.startAsync()啟動異步處理 返回一個異步上下文對象AsyncContext最終是使用該上下文對象來進行異步業(yè)務(wù)邏輯處理,其中有兩個核心方法
asyncContext.start(new WeekTask(asyncContext)); 添加一個異步任務(wù)該任務(wù)是一個Runnable線程接口,這里就清晰了其實是servlet線程將處理任務(wù)交給另一個子線程,servlet直接返回從而達到提高系統(tǒng)吞吐量的作用。
對于異步請求可以我們需要獲取其中的結(jié)果,所有這里提供了監(jiān)聽器模式添加事件監(jiān)聽AsyncListener
| onComplete | 異步請求處理完成觸發(fā) 前提示需要調(diào)用 asyncContext.complete()方法(因為程序也不知道什么時候任務(wù)算是調(diào)用完畢了) |
| onTimeout | 異步請求處理超時觸發(fā),一般來說采用異步請求的任務(wù)都是比較耗時的任務(wù),所以需要修改servlet默認的超時時間(修改的長一點) |
| onError | 異步處理錯誤的時候觸發(fā) |
| onStartAsync | 異步處理開始的時候觸發(fā)即為request.startAsync(),因為添加監(jiān)聽器在startAsync()方法后,所以第一個啟動是無法觸發(fā)該監(jiān)聽的 |
這里異步處理只是簡單的打印了相關(guān)日志,不過真實的業(yè)務(wù)場景中可以寫復(fù)雜的業(yè)務(wù)處理邏輯。
2.3、異步任務(wù)
這里提供相關(guān)的異步操作是實現(xiàn)runnable的線程實現(xiàn)類,同時這里提供了相關(guān)Job,PmJob(產(chǎn)品任務(wù)),RDJob(研發(fā)任務(wù)),TestJob(測試任務(wù)),每個任務(wù)模擬了10秒的耗時任務(wù)。
/**
* @ClassName WeekTask
* @Desc 每周任務(wù)
* @Author xieqx
* @Date 2020/12/10 9:36
**/
public class WeekTask implements Runnable {
private List<Job> jobs = null;
private AsyncContext asyncContext = null;
//這里初始化產(chǎn)品任務(wù)PmJob、研發(fā)任務(wù)RDJob 測試任務(wù)TestJob
public WeekTask(AsyncContext asyncContext) {
this.asyncContext = asyncContext;
jobs = new ArrayList<>();
PmJob pmJob = new PmJob();
RDJob rdJob = new RDJob();
TestJob testJob = new TestJob();
jobs.add(pmJob);
jobs.add(rdJob);
jobs.add(testJob);
}
@Override
public void run() {
for(Job job:jobs){
job.execute();
}
System.out.println("周任務(wù)工作完成");
//job執(zhí)行完成后通知
asyncContext.complete();
}
}PmJob
/**
* @ClassName PmTask
* @Desc 產(chǎn)品經(jīng)理任務(wù)
* @Author xieqx
* @Date 2020/12/9 16:03
**/
public class PmJob implements Job {
@Override
public void execute() {
System.out.println("產(chǎn)品經(jīng)理開評審會議");
try {
Thread.sleep(10);
System.out.println("模擬需求評審會議...");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}RDJob
/**
* @ClassName PmTask
* @Desc 研發(fā)任務(wù)
* @Author xieqx
* @Date 2020/12/9 16:03
**/
public class RDJob implements Job {
@Override
public void execute() {
System.out.println("程序猿開始開發(fā)");
try {
Thread.sleep(10);
System.out.println("程序猿哼哧哼哧干活中...");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}TestJob
/**
* @ClassName TestJob
* @Desc 測試任務(wù)
* @Author xieqx
* @Date 2020/12/9 16:03
**/
public class TestJob implements Job {
@Override
public void execute() {
System.out.println("測試開始測試");
try {
Thread.sleep(10);
System.out.println("測試用例測試...");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}2.4、測試場景
請求立馬響應(yīng),但是異步任務(wù)在后面處理


到此這篇關(guān)于servlet異步請求的實現(xiàn)的文章就介紹到這了,更多相關(guān)servlet異步請求內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
使用Java對數(shù)據(jù)庫進行基本的查詢和更新操作
這篇文章主要介紹了使用Java對數(shù)據(jù)庫進行基本的查詢和更新操作,是Java入門學(xué)習(xí)中的基礎(chǔ)知識,需要的朋友可以參考下2015-10-10
Java中的NoClassDefFoundError報錯含義解析
這篇文章主要為大家介紹了Java中的NoClassDefFoundError含義詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助2023-11-11
MyBatis3傳遞多個參數(shù)(Multiple Parameters)
這篇文章主要介紹了MyBatis3傳遞多個參數(shù),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07
SpringBoot中yml多環(huán)境配置的3種方法
這篇文章主要給大家介紹了SpringBoot中yml多環(huán)境配置的3種方法,文中有詳細的代碼示例供大家參考,對大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2023-10-10
Spring中11個最常用的擴展點總結(jié),你知道幾個
我們知道IOC(控制反轉(zhuǎn))和AOP(面向切面編程)是spring的基石,除此之外spring的擴展能力非常強,下面這篇文章主要給大家介紹了關(guān)于Spring中11個最常用的擴展點的相關(guān)資料,需要的朋友可以參考下2022-12-12

