1. 程式人生 > >Servlet中的過濾器Filter詳解

Servlet中的過濾器Filter詳解

rep 釋放 ons smo text 執行c 總結 throws 程序啟動

轉自: http://blog.csdn.net/sd0902/article/details/8395641

web.xml中元素執行的順序listener->filter->struts攔截器->servlet。

1.過濾器的概念

Java中的Filter 並不是一個標準的Servlet ,它不能處理用戶請求,也不能對客戶端生成響應。 主要用於對HttpServletRequest 進行預處理,也可以對HttpServletResponse 進行後處理,是個典型的處理鏈。

優點:過濾鏈的好處是,執行過程中任何時候都可以打斷,只要不執行chain.doFilter()就不會再執行後面的過濾器和請求的內容。而在實際使用時,就要特別註意過濾鏈的執行順序問題

2.過濾器的作用描述

  • 在HttpServletRequest 到達Servlet 之前,攔截客戶的HttpServletRequest 。
  • 根據需要檢查HttpServletRequest ,也可以修改HttpServletRequest 頭和數據。
  • 在HttpServletResponse 到達客戶端之前,攔截HttpServletResponse 。
  • 根據需要檢查HttpServletResponse ,可以修改HttpServletResponse 頭和數據。

3.過濾器的執行流程

技術分享技術分享


4.Filter接口

1.如何驅動

在 web 應用程序啟動時,web 服務器將根據 web.xml 文件中的配置信息來創建每個註冊的 Filter 實例對象,並將其保存在服務器的內存中

2.方法介紹

    • init() Init 方法在 Filter 生命周期中僅執行一次,web 容器在調用 init 方法時
    • destory() 在Web容器卸載 Filter 對象之前被調用。該方法在Filter的生命周期中僅執行一次。在這個方法中,可以釋放過濾器使用的資源。
    • doFilter() Filter 鏈的執行

5.FilterChain接口

1.如何實例化

代表當前 Filter 鏈的對象。由容器實現,容器將其實例作為參數傳入過濾器對象的doFilter()方法中

2.作用

調用過濾器鏈中的下一個過濾器

filter實例:

web.xml配置

[html] view plaincopy
  1. <!-- 編碼過濾器 -->
  2. <filter>
  3. <filter-name>setCharacterEncoding</filter-name>
  4. <filter-class>com.company.strutstudy.web.servletstudy.filter.EncodingFilter</filter-class>
  5. <init-param>
  6. <param-name>encoding</param-name>
  7. <param-value>utf-8</param-value>
  8. </init-param>
  9. </filter>
  10. <filter-mapping>
  11. <filter-name>setCharacterEncoding</filter-name>
  12. <url-pattern>/*</url-pattern>
  13. </filter-mapping>
  14. <!-- 請求url日誌記錄過濾器 -->
  15. <filter>
  16. <filter-name>logfilter</filter-name>
  17. <filter-class>com.company.strutstudy.web.servletstudy.filter.LogFilter</filter-class>
  18. </filter>
  19. <filter-mapping>
  20. <filter-name>logfilter</filter-name>
  21. <url-pattern>/*</url-pattern>
  22. </filter-mapping>


編碼攔截器:

[java] view plaincopy
  1. public class EncodingFilter implements Filter {
  2. private String encoding;
  3. private Map<String, String> params = new HashMap<String, String>();
  4. // 項目結束時就已經進行銷毀
  5. public void destroy() {
  6. System.out.println("end do the encoding filter!");
  7. params=null;
  8. encoding=null;
  9. }
  10. public void doFilter(ServletRequest req, ServletResponse resp,
  11. FilterChain chain) throws IOException, ServletException {
  12. //UtilTimerStack.push("EncodingFilter_doFilter:");
  13. System.out.println("before encoding " + encoding + " filter!");
  14. req.setCharacterEncoding(encoding);
  15. // resp.setCharacterEncoding(encoding);
  16. // resp.setContentType("text/html;charset="+encoding);
  17. chain.doFilter(req, resp);
  18. System.out.println("after encoding " + encoding + " filter!");
  19. System.err.println("----------------------------------------");
  20. //UtilTimerStack.pop("EncodingFilter_doFilter:");
  21. }
  22. // 項目啟動時就已經進行讀取
  23. public void init(FilterConfig config) throws ServletException {
  24. System.out.println("begin do the encoding filter!");
  25. encoding = config.getInitParameter("encoding");
  26. for (Enumeration e = config.getInitParameterNames(); e
  27. .hasMoreElements();) {
  28. String name = (String) e.nextElement();
  29. String value = config.getInitParameter(name);
  30. params.put(name, value);
  31. }
  32. }
  33. }

日誌攔截器:

[java] view plaincopy
  1. public class LogFilter implements Filter {
  2. FilterConfig config;
  3. public void destroy() {
  4. this.config = null;
  5. }
  6. public void doFilter(ServletRequest req, ServletResponse res,
  7. FilterChain chain) throws IOException, ServletException {
  8. // 獲取ServletContext 對象,用於記錄日誌
  9. ServletContext context = this.config.getServletContext();
  10. //long before = System.currentTimeMillis();
  11. System.out.println("before the log filter!");
  12. //context.log("開始過濾");
  13. // 將請求轉換成HttpServletRequest 請求
  14. HttpServletRequest hreq = (HttpServletRequest) req;
  15. // 記錄日誌
  16. System.out.println("Log Filter已經截獲到用戶的請求的地址:"+hreq.getServletPath() );
  17. //context.log("Filter已經截獲到用戶的請求的地址: " + hreq.getServletPath());
  18. try {
  19. // Filter 只是鏈式處理,請求依然轉發到目的地址。
  20. chain.doFilter(req, res);
  21. } catch (Exception e) {
  22. e.printStackTrace();
  23. }
  24. System.out.println("after the log filter!");
  25. //long after = System.currentTimeMillis();
  26. // 記錄日誌
  27. //context.log("過濾結束");
  28. // 再次記錄日誌
  29. //context.log(" 請求被定位到" + ((HttpServletRequest) req).getRequestURI()
  30. // + "所花的時間為: " + (after - before));
  31. }
  32. public void init(FilterConfig config) throws ServletException {
  33. System.out.println("begin do the log filter!");
  34. this.config = config;
  35. }
  36. }

HelloServlet類:

[java] view plaincopy
  1. public class HelloWorldServlet extends HttpServlet{
  2. /**
  3. * 查看httpservlet實現的service一看便知,起到了一個controll控制器的作用(轉向的)
  4. * 之後便是跳轉至我們熟悉的doget,dopost等方法中
  5. */
  6. @Override
  7. protected void service(HttpServletRequest req, HttpServletResponse resp)
  8. throws ServletException, IOException {
  9. System.out.println("doservice..."+this.getInitParameter("encoding"));
  10. super.service(req, resp);
  11. }
  12. @Override
  13. protected void doGet(HttpServletRequest req, HttpServletResponse resp)
  14. throws ServletException, IOException {
  15. System.out.println("doget...");
  16. doPost(req, resp);
  17. }
  18. @Override
  19. protected void doPost(HttpServletRequest req, HttpServletResponse resp)
  20. throws ServletException, IOException {
  21. System.out.println("dopost...");
  22. }
  23. }

結果:

[plain] view plaincopy
  1. before encoding utf-8 filter!
  2. before the log filter!
  3. Log Filter已經截獲到用戶的請求的地址:/hello
  4. doservice...UTF-8
  5. doget...
  6. dopost...
  7. after the log filter!
  8. after encoding utf-8 filter!
  9. ----------------------------------------

總結:

1.過濾器執行流程

2.常用過濾器

[html] view plaincopy
    1. <pre name="code" class="plain"><pre></pre><pre name="code" class="plain"></pre><pre></pre>
    2. <pre></pre>
    3. <pre></pre>
    4. <pre></pre>
    5. <pre></pre>
    6. <pre></pre>
    7. <pre></pre>
    8. </pre>

Servlet中的過濾器Filter詳解