IT源码网

filter执行顺序讲解

luoye 2021年04月03日 程序员 126 0

Filter可以担任浏览器与JSP/Servlet之间的一个中介处理者,一些request的前置处理动作及一些response的后置处理,都可以交由这个中介处理者来完成,当然Filter可以达到的功能,也可以直接撰写在JSP/Servlet中,但如果有很多JSP/Servlet都需要相同的处理,例如某些网页都需要统一的身份验证方式时,与其在每一个网页中都撰写验证的程式码,不如直接撰写Filter,让它来统一进行处理。

Filter实际上是一个纯綷的Java类别程式,它要实作javax.servlet.Filter介面,这个介面中有三个必须实作的方法:init()、destory()与doFilter()。init()是Filter类别被载入时会执行的方法,而destory()是 Filter物件生命週期结束时会执行的方,至于doFilter()则是实作Filter功能的核心,想要Filter完成的工作就撰写在其中,先来撰写一个简单的Filter类,如下:
    
 

import java.io. * ; 
import javax.servlet. * ; 
import javax.servlet.http. * ; 
public class FilterDemo1 implements Filter { 
        public void init(FilterConfig config) throws ServletException {} 
        public void destroy() {} 
        public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, 
        ServletException { 
                PrintWriter out = response.getWriter(); 
                out.println("Filter 1 is doing!!!"); 
                chain.doFilter(request, response); 
                out.println("Filter 1's work is done"); 
        } 
}

在这个程序中,只是简单的显示一些文字,表示Filter的doFilter()方法有被执行,doFilter()传入三个参数,ServletRequest、ServletResponse、FilterChain,可以获得request与response物件,以对它们进行一些处理,像是设定标头、编码格式或包装压缩功能等等,至于FilterChain则是关于Filter执行顺序的物件,这个顺序是设定在 web.xml中,必须呼叫它的doFilter()方法,才可以执行下一个Filter,如果是最后一个Filter,则执行客户端所请求的 JSP/Servlet网页。
为了要能使用Filter,必须在web.xml中撰写一些设定,一个设定例子如下:

<filter>  
   <filter-name> 
    FilterDemo1 
   </filter-name>  
   <filter-class> 
    onlyfun.caterpillar.FilterDemo1 
   </filter-class>  
  </filter>  
  <filter-mapping>  
   <filter-name> 
    FilterDemo1 
   </filter-name>  
   <url-pattern> 
    /* 
   </url-pattern>  
  </filter-mapping>  



<filter>与</filter>之间撰写Filter的名称与实际载入的类别,Filter类别档是放置在WEB-INF/classes/下,而<filter-mapping>与</filter-mapping>之间则撰写Filter名称与所要过滤的资源对象,在这边设定为/*,表示请求这个Web应用程式下的所有资源,都要先经过 Filter的处理。

如果设定的<url-pattern>套用了两个Filter,则FilterChain在执行doFilter()方法时,其调用的Filter顺序就是您在web.xml中撰写Filter的顺序,例如:
 

import java.io.*; 
 
import javax.servlet.*; 
import javax.servlet.http.*; 
 
 
public class FilterDemo2 implements Filter { 
    public void init(FilterConfig config) throws ServletException { 
    } 
 
    public void destroy() { 
    } 
 
    public void doFilter(ServletRequest request, ServletResponse response, 
        FilterChain chain) throws IOException, ServletException { 
        PrintWriter out = response.getWriter(); 
        out.println("Filter 2 is doing!!!"); 
        chain.doFilter(request, response); 
        out.println("Filter 2's work is done"); 
    } 
} 


web.xml中设定的内容是:
 

<filter>  
   <filter-name> 
    FilterDemo1 
   </filter-name>  
   <filter-class> 
    onlyfun.caterpillar.FilterDemo1 
   </filter-class>  
  </filter>  
  <filter-mapping>  
   <filter-name> 
    FilterDemo1 
   </filter-name>  
   <url-pattern> 
    /* 
   </url-pattern>  
  </filter-mapping>  
  <filter>  
   <filter-name> 
    FilterDemo2 
   </filter-name>  
   <filter-class> 
    onlyfun.caterpillar.FilterDemo2 
   </filter-class>  
  </filter>  
  <filter-mapping>  
   <filter-name> 
    FilterDemo2 
   </filter-name>  
   <url-pattern> 
    /* 
   </url-pattern>  
  </filter-mapping>  

则请求Web应用下的任一个资源,都会先执行FilterDemo1,然后再执行FilterDemo2,最后调用请求的资源,如果请求的资源是 JSP网页,而它的功能只是印出 5 个Hello! World!:
    
<% 
for(int i = 0; i < 5; i++) 
out.println("Hello! World!"); 
%>


所得到的结果是:
Filter 1 is doing!!!
Filter 2 is doing!!!
Hello! World!
Hello! World!
Hello! World!
Hello! World!
Hello! World!
Filter 2's work is done
Filter 1's work is done

发布评论

分享到:

IT源码网

微信公众号号:IT虾米 (左侧二维码扫一扫)欢迎添加!

getRequestDispatcher()与sendRedirect()的区别讲解
你是第一个吃螃蟹的人
发表评论

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。