欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Servlet Filter過(guò)濾器執(zhí)行順序

 更新時(shí)間:2020年12月01日 14:23:26   作者:DayRain  
這篇文章主要介紹了Servlet Filter過(guò)濾器執(zhí)行順序的相關(guān)資料,幫助大家更好的理解為什么要用過(guò)濾器,感興趣的朋友可以了解下

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)文章!

相關(guān)文章

  • 解析Spring Mvc Long類型精度丟失問(wè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代理

    詳解如何通過(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-08
  • Spring Boot 中的 @PutMapping 注解原理及使用小結(jié)

    Spring Boot 中的 @PutMapping 注解原理及使用小結(jié)

    在本文中,我們介紹了 Spring Boot 中的 @PutMapping 注解,它可以將 HTTP PUT 請(qǐng)求映射到指定的處理方法上,我們還介紹了 @PutMapping 注解的原理以及如何在 Spring Boot 中使用它,感興趣的朋友跟隨小編一起看看吧
    2023-12-12
  • JavaWeb實(shí)現(xiàn)用戶登錄與注冊(cè)功能

    JavaWeb實(shí)現(xiàn)用戶登錄與注冊(cè)功能

    這篇文章主要為大家詳細(xì)介紹了JavaWeb實(shí)現(xiàn)用戶登錄與注冊(cè)功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • Java中三種簡(jiǎn)單注解介紹和代碼實(shí)例

    Java中三種簡(jiǎn)單注解介紹和代碼實(shí)例

    這篇文章主要介紹了Java中三種簡(jiǎn)單注解介紹和代碼實(shí)例,本文講解了Override注解、Deprecated注解、Suppresswarnings注解、元注解等內(nèi)容,需要的朋友可以參考下
    2014-09-09
  • Java Socket+mysql實(shí)現(xiàn)簡(jiǎn)易文件上傳器的代碼

    Java 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-10
  • java解析任意層數(shù)json字符串的方法

    java解析任意層數(shù)json字符串的方法

    一個(gè)方法解析任意層數(shù)的json字符竄:使用正則表達(dá)式,遞歸算法,將jsonArray解析出后添加到List, JsonObject添加至Map
    2014-02-02
  • Eclipse+Java+Swing實(shí)現(xiàn)圖書管理系統(tǒng)(詳細(xì)代碼)

    Eclipse+Java+Swing實(shí)現(xiàn)圖書管理系統(tǒng)(詳細(xì)代碼)

    這篇文章主要介紹了Eclipse+Java+Swing實(shí)現(xiàn)圖書管理系統(tǒng)并附上詳細(xì)代碼,需要的小伙伴可以參考一下,希望對(duì)你有所幫助
    2022-01-01
  • Spring Security動(dòng)態(tài)權(quán)限的實(shí)現(xiàn)方法詳解

    Spring 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-06
  • Java ScheduledExecutorService的具體使用

    Java ScheduledExecutorService的具體使用

    ScheduledExecutorService有線程池的特性,也可以實(shí)現(xiàn)任務(wù)循環(huán)執(zhí)行,本文主要介紹了Java ScheduledExecutorService的具體使用,具有一定的參考價(jià)值,感興趣的可以了解一下
    2023-05-05

最新評(píng)論