Servlet Filter過(guò)濾器執(zhí)行順序
Servlet中的過(guò)濾器相當(dāng)于守護(hù)后臺(tái)資源的一道關(guān)卡,我們可以在過(guò)濾器中進(jìn)行身份校驗(yàn)、權(quán)限認(rèn)證、請(qǐng)求過(guò)濾等。
過(guò)濾器本身并不難,我們只需要知道他的定義方法、作用范圍、執(zhí)行順序即可。
網(wǎng)上對(duì)于過(guò)濾器執(zhí)行順序的描述可能會(huì)讓人產(chǎn)生誤解。
圖片來(lái)源于網(wǎng)絡(luò)
客戶端請(qǐng)求到達(dá)的時(shí)候,經(jīng)過(guò)一次過(guò)濾器。
服務(wù)器處理完請(qǐng)求的時(shí)候,經(jīng)過(guò)一次過(guò)濾器。
雖然經(jīng)過(guò)兩次過(guò)濾器,但不代表同樣的代碼執(zhí)行了兩次。
下面做了個(gè)簡(jiǎn)單的測(cè)試,看下執(zhí)行結(jié)果就應(yīng)該知道真正的執(zhí)行流程了。
測(cè)試環(huán)境
tomcat9(servlet4.0)
jdk1.8
新版servlet可以通過(guò)注解注冊(cè)servlet組件以及過(guò)濾器,無(wú)需再到web.xml下注冊(cè)了。
測(cè)試過(guò)程
測(cè)試之間要先知道filterChain(過(guò)濾鏈)是干嘛的。
一個(gè)過(guò)濾器處理完后,會(huì)把request和response對(duì)象通過(guò)filterchain傳遞給下一個(gè)過(guò)濾器,如果沒(méi)有下一個(gè)過(guò)濾器,則會(huì)直接開始執(zhí)行業(yè)務(wù)代碼,
單個(gè)過(guò)濾器
定義一個(gè)過(guò)濾器A
@WebFilter(value = "/*", filterName="A") public class FilterA implements Filter { @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); System.out.println(format.format(new Date())); System.out.println("A:攔截1"); chain.doFilter(request, response); System.out.println(format.format(new Date())); System.out.println("A:攔截2"); } @Override public void init(FilterConfig filterConfig) throws ServletException { } @Override public void destroy() { } }
定義一個(gè)servlet,sleep5秒
@WebServlet("/mainUrl") public class MainController extends HttpServlet { private static final long serialVersionUID = 1L; /** * @see HttpServlet#HttpServlet() */ public MainController() { super(); // TODO Auto-generated constructor stub } /** * @see HttpServlet#doGet(HttpServletRequest request, HttpServletResponse response) */ protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // TODO Auto-generated method stub try { Thread.sleep(5000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } } /** * @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse response) */ protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // TODO Auto-generated method stub doGet(request, response); } }
運(yùn)行結(jié)果
2020-12-01 10:46:50 A:攔截1 2020-12-01 10:46:55 A:攔截2
執(zhí)行順序:
filterChain之前的代碼 ——>業(yè)務(wù)處理——>filterChain之后的代碼。
多個(gè)過(guò)濾器
servlet的注解在多個(gè)過(guò)濾器的情況下,是按照過(guò)濾器的名稱來(lái)排序的,例如A開頭的過(guò)濾器,在B開頭的后面。
A過(guò)濾器
@WebFilter(value = "/*", filterName="A") public class FilterA implements Filter { @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); System.out.println(format.format(new Date())); System.out.println("A:攔截1"); chain.doFilter(request, response); System.out.println(format.format(new Date())); System.out.println("A:攔截2"); } @Override public void init(FilterConfig filterConfig) throws ServletException { } @Override public void destroy() { } }
B過(guò)濾器
@WebFilter(value = "/*", filterName="B") public class FilterB implements Filter{ @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { // TODO Auto-generated method stub SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); System.out.println(format.format(new Date())); System.out.println("B:攔截1"); chain.doFilter(request, response); System.out.println(format.format(new Date())); response.setContentType("normal content"); System.out.println("B:攔截2"); } @Override public void init(FilterConfig filterConfig) throws ServletException { } @Override public void destroy() { } }
運(yùn)行結(jié)果:
2020-12-01 10:53:00 A:攔截1 2020-12-01 10:53:00 B:攔截1 2020-12-01 10:53:05 B:攔截2 2020-12-01 10:53:05 A:攔截2
執(zhí)行順序:
B:攔截1和B:攔截2之間,停頓了5秒處理業(yè)務(wù)。所以先執(zhí)行了 chain.doFilter前的 A、B過(guò)濾器代碼,處理完業(yè)務(wù)返回的時(shí)候正好相反,先返回執(zhí)行B的代碼,再執(zhí)行的A的代碼。
總結(jié)
再來(lái)看這個(gè)圖,可以略微改一下了。
分界線是filterChain過(guò)濾鏈,請(qǐng)求進(jìn)來(lái)的時(shí)候,執(zhí)行filterchain之前的代碼,返回response的時(shí)候,執(zhí)行filterchain之后的代碼。
多個(gè)過(guò)濾器之間的執(zhí)行順序,滿足“先進(jìn)后出” (棧結(jié)構(gòu))的原則。
其他
如果在測(cè)試過(guò)程中,發(fā)現(xiàn)過(guò)濾器執(zhí)行了很多次,那么也可能是因?yàn)闇y(cè)試環(huán)境中包含了某些靜態(tài)資源。
過(guò)濾器A
@WebFilter(value = "/*", filterName="A") public class FilterA implements Filter { @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); System.out.println(format.format(new Date())); System.out.println("A:攔截1"); chain.doFilter(request, response); System.out.println(format.format(new Date())); System.out.println("A:攔截2"); } @Override public void init(FilterConfig filterConfig) throws ServletException { } @Override public void destroy() { } }
過(guò)濾器B
@WebFilter(value = "/*", filterName="B") public class FilterB implements Filter{ @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { // TODO Auto-generated method stub HttpServletRequest req = (HttpServletRequest) request; SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); System.out.println(req.getRequestURL()); System.out.println(format.format(new Date())); System.out.println("B:攔截1"); chain.doFilter(request, response); System.out.println(format.format(new Date())); response.setContentType("normal content"); System.out.println("B:攔截2"); } @Override public void init(FilterConfig filterConfig) throws ServletException { } @Override public void destroy() { } }
主程序
@WebServlet("/mainUrl") public class MainController extends HttpServlet { private static final long serialVersionUID = 1L; /** * @see HttpServlet#HttpServlet() */ public MainController() { super(); // TODO Auto-generated constructor stub } /** * @see HttpServlet#doGet(HttpServletRequest request, HttpServletResponse response) */ protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // TODO Auto-generated method stub try { Thread.sleep(5000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } request.getRequestDispatcher("/WEB-INF/pages/main.jsp").forward(request, response); // response.sendRedirect("/WEB-INF/pages/main.jsp"); } /** * @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse response) */ protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // TODO Auto-generated method stub doGet(request, response); } }
執(zhí)行結(jié)果:
2020-12-01 11:09:38 A:攔截1 http://localhost:8080/StudentManage/mainUrl 2020-12-01 11:09:38 B:攔截1 2020-12-01 11:09:43 B:攔截2 2020-12-01 11:09:43 A:攔截2 2020-12-01 11:09:44 A:攔截1 http://localhost:8080/StudentManage/css/bootstrap.css.map 2020-12-01 11:09:44 B:攔截1 2020-12-01 11:09:44 B:攔截2 2020-12-01 11:09:44 A:攔截2
轉(zhuǎn)發(fā)(forward)的頁(yè)面中需要請(qǐng)求靜態(tài)資源,再次觸發(fā)了過(guò)濾器。
以上就是Servlet Filter過(guò)濾器執(zhí)行順序的詳細(xì)內(nèi)容,更多關(guān)于Servlet Filter過(guò)濾器的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- 淺談SpringMVC的攔截器(Interceptor)和Servlet 的過(guò)濾器(Filter)的區(qū)別與聯(lián)系 及SpringMVC 的配置文件
- JavaWeb Servlet中Filter過(guò)濾器的詳解
- servlet過(guò)濾器(Filter)詳解(九)
- 詳解Servlet之過(guò)濾器(Filter)
- Spring Boot使用Servlet及Filter過(guò)程詳解
- springboot掃描自定義的servlet和filter代碼詳解
- SpringBoot初始教程之Servlet、Filter、Listener配置詳解
- 詳談Servlet和Filter的區(qū)別以及兩者在Struts2和Springmvc中的應(yīng)用
- web.xml中servlet, bean, filter, listenr 加載順序_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
- Filter、Servlet、Listener的學(xué)習(xí)_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
- Spring Boot 編寫Servlet、Filter、Listener、Interceptor的方法
- 解決cannot be cast to javax.servlet.Filter 報(bào)錯(cuò)的問(wèn)題
相關(guān)文章
解析Spring Mvc Long類型精度丟失問(wèn)題
在平時(shí)開發(fā)過(guò)程中,經(jīng)常會(huì)使用long類型作為id的類型,但是在使用過(guò)程中會(huì)導(dǎo)致long類型數(shù)據(jù)轉(zhuǎn)換為number類型時(shí)的后兩位變?yōu)?,今天小編給大家分享Spring Mvc Long類型精度丟失問(wèn)題,需要的朋友參考下吧2021-06-06詳解如何通過(guò)Java實(shí)現(xiàn)類似Nginx代理
最近遇到一個(gè)問(wèn)題,在內(nèi)網(wǎng)環(huán)境中部署的項(xiàng)目需要調(diào)用外網(wǎng)完成一些應(yīng)用,一般情況我們可以通過(guò)增加一臺(tái)機(jī)器,部署到可以訪問(wèn)外網(wǎng)的服務(wù)器上,然后內(nèi)網(wǎng)直接連接該機(jī)器通過(guò)Nginx進(jìn)行代理即可,所以本文介紹了如何通過(guò)Java實(shí)現(xiàn)類似Nginx代理,需要的朋友可以參考下2024-08-08Spring Boot 中的 @PutMapping 注解原理及使用小結(jié)
在本文中,我們介紹了 Spring Boot 中的 @PutMapping 注解,它可以將 HTTP PUT 請(qǐng)求映射到指定的處理方法上,我們還介紹了 @PutMapping 注解的原理以及如何在 Spring Boot 中使用它,感興趣的朋友跟隨小編一起看看吧2023-12-12JavaWeb實(shí)現(xiàn)用戶登錄與注冊(cè)功能
這篇文章主要為大家詳細(xì)介紹了JavaWeb實(shí)現(xiàn)用戶登錄與注冊(cè)功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-08-08Java Socket+mysql實(shí)現(xiàn)簡(jiǎn)易文件上傳器的代碼
最近在做一個(gè)小項(xiàng)目,項(xiàng)目主要需求是實(shí)現(xiàn)一個(gè)文件上傳器,通過(guò)客戶端的登陸,把本地文件上傳到服務(wù)器的數(shù)據(jù)庫(kù)(本地的)。下面通過(guò)本文給大家分享下實(shí)現(xiàn)代碼,感興趣的朋友一起看看吧2016-10-10Eclipse+Java+Swing實(shí)現(xiàn)圖書管理系統(tǒng)(詳細(xì)代碼)
這篇文章主要介紹了Eclipse+Java+Swing實(shí)現(xiàn)圖書管理系統(tǒng)并附上詳細(xì)代碼,需要的小伙伴可以參考一下,希望對(duì)你有所幫助2022-01-01Spring Security動(dòng)態(tài)權(quán)限的實(shí)現(xiàn)方法詳解
這篇文章主要和小伙伴們簡(jiǎn)單介紹下 Spring Security 中的動(dòng)態(tài)權(quán)限方案,以便于小伙伴們更好的理解 TienChin 項(xiàng)目中的權(quán)限方案,感興趣的可以了解一下2022-06-06Java ScheduledExecutorService的具體使用
ScheduledExecutorService有線程池的特性,也可以實(shí)現(xiàn)任務(wù)循環(huán)執(zhí)行,本文主要介紹了Java ScheduledExecutorService的具體使用,具有一定的參考價(jià)值,感興趣的可以了解一下2023-05-05