Filter簡介
Filter也稱之為過濾器,它是Servlet技術中最激動人心的技術,WEB開發人員通過Filter技術,對web伺服器管理的所有web資源:例如Jsp, Servlet, 靜态圖檔檔案或靜态 html 檔案等進行攔截,進而實作一些特殊的功能。例如實作URL級别的權限通路控制、過濾敏感詞彙、壓縮響應資訊等一些進階功能
Servlet API中提供了一個Filter接口,開發web應用時,如果編寫的Java類實作了這個接口,則把這個java類稱之為過濾器Filter。通過Filter技術,開發人員可以實作使用者在通路某個目标資源之前,對通路的請求和響應進行攔截
Filter是如何實作攔截的?
Filter接口中有一個doFilter方法,當開發人員編寫好Filter,并配置對哪個web資源進行攔截後,WEB伺服器每次在調用web資源的service方法之前,都會先調用一下filter的doFilter方法,是以,在該方法内編寫代碼可達到如下目的:
1、調用目标資源之前,讓一段代碼執行
2、是否調用目标資源(即是否讓使用者通路web資源)。
3、web伺服器在調用doFilter方法時,會傳遞一個filterChain對象進來,filterChain對象是filter接口中最重要的一個對象,它也提供了一個doFilter方法,開發人員可以根據需求決定是否調用此方法,調用該方法,則web伺服器就會調用web資源的service方法,即web資源就會被通路,否則web資源不會被通路。
Filter開發分為二個步驟:
編寫java類實作Filter接口,并實作其doFilter方法。
在 web.xml 檔案中使用<filter>和<filter-mapping>元素對編寫的filter類進行注冊,并設定它所能攔截的資源。
Filter鍊
在一個web應用中,可以開發編寫多個Filter,這些Filter組合起來稱之為一個Filter鍊。
web伺服器根據Filter在web.xml檔案中的注冊順序,決定先調用哪個Filter,當第一個Filter的doFilter方法被調用時,web伺服器會建立一個代表Filter鍊的FilterChain對象傳遞給該方法。在doFilter方法中,開發人員如果調用了FilterChain對象的doFilter方法,則web伺服器會檢查FilterChain對象中是否還有filter,如果有,則調用第2個filter,如果沒有,則調用目标資源。
Filter的生命周期
init(FilterConfig filterConfig) throws ServletException:
和我們編寫的Servlet程式一樣,Filter的建立和銷毀由WEB伺服器負責。 web 應用程式啟動時,web 伺服器将建立Filter 的執行個體對象,并調用其init方法,完成對象的初始化功能,進而為後續的使用者請求作好攔截的準備工作(注:filter對象隻會建立一次,init方法也隻會執行一次。示例 )
開發人員通過init方法的參數,可獲得代表目前filter配置資訊的FilterConfig對象。(filterConfig對象見下頁PPT)
destroy():
在Web容器解除安裝 Filter 對象之前被調用。該方法在Filter的生命周期中僅執行一次。在這個方法中,可以釋放過濾器使用的資源
FilterConfig接口
使用者在配置filter時,可以使用<init-param>為filter配置一些初始化參數,當web容器執行個體化Filter對象,調用其init方法時,會把封裝了filter初始化參數的filterConfig對象傳遞進來。是以開發人員在編寫filter時,通過filterConfig對象的方法,就可獲得:
String getFilterName():得到filter的名稱。
String getInitParameter(String name): 傳回在部署描述中指定名稱的初始化參數的值。如果不存在傳回null.
Enumeration getInitParameterNames():傳回過濾器的所有初始化參數的名字的枚舉集合。
public ServletContext getServletContext():傳回Servlet上下文對象的引用。
Filter的部署
注冊Filter
<filter>
<filter-name>testFitler</filter-name>
<filter-class>org.test.TestFiter</filter-class>
<init-param>
<param-name>word_file</param-name>
<param-value>/WEB-INF/word.txt</param-value>
</init-param>
</filter>
<filter-name>用于為過濾器指定一個名字,該元素的内容不能為空。
<filter-class>元素用于指定過濾器的完整的限定類名。
<init-param>元素用于為過濾器指定初始化參數,它的子元素<param-name>指定參數的名字,<param-value>指定參數的值。在過濾器中,可以使用FilterConfig接口對象來通路初始化參數。
映射Filter
<filter-mapping>
<filter-name>testFilter</filter-name>
<url-pattern>/test.jsp</url-pattern>
</filter-mapping>
<filter-mapping>
<filter-name>testFilter</filter-name>
<url-pattern>/index.jsp</url-pattern>
<dispatcher>REQUEST</dispatcher>
<dispatcher>FORWARD</dispatcher>
</filter-mapping>
<filter-mapping>元素用于設定一個 Filter 所負責攔截的資源。一個Filter攔截的資源可通過兩種方式來指定:Servlet 名稱和資源通路的請求路徑
<filter-name>子元素用于設定filter的注冊名稱。該值必須是在<filter>元素中聲明過的過濾器的名字
<url-pattern>設定 filter 所攔截的請求路徑(過濾器關聯的URL樣式)
<servlet-name>指定過濾器所攔截的Servlet名稱。
<dispatcher>指定過濾器所攔截的資源被 Servlet 容器調用的方式,可以是REQUEST,INCLUDE,FORWARD和ERROR之一,預設REQUEST。使用者可以設定多個<dispatcher> 子元素用來指定 Filter 對資源的多種調用方式進行攔截。
<dispatcher> 子元素可以設定的值及其意義:
REQUEST:當使用者直接通路頁面時,Web容器将會調用過濾器。如果目标資源是通過RequestDispatcher的include()或forward()方法通路時,那麼該過濾器就不會被調用。
INCLUDE:如果目标資源是通過RequestDispatcher的include()方法通路時,那麼該過濾器将被調用。除此之外,該過濾器不會被調用。
FORWARD:如果目标資源是通過RequestDispatcher的forward()方法通路時,那麼該過濾器将被調用,除此之外,該過濾器不會被調用。
ERROR:如果目标資源是通過聲明式異常處理機制調用時,那麼該過濾器将被調用。除此之外,過濾器不會被調用。
filter執行個體-統一全站字元編碼的過濾器
1、建立一個類webFilter 并實作接口 Filter
1 package fnz.filter;
2 import java.io.IOException;
3 import java.lang.reflect.InvocationHandler;
4 import java.lang.reflect.Method;
5 import java.lang.reflect.Proxy;
6
7 import javax.servlet.Filter;
8 import javax.servlet.FilterChain;
9 import javax.servlet.FilterConfig;
10 import javax.servlet.ServletException;
11 import javax.servlet.ServletRequest;
12 import javax.servlet.ServletResponse;
13 import javax.servlet.http.HttpServletRequest;
14 import javax.servlet.http.HttpServletResponse;
15 public class webFilter implements Filter {
16
17 public void init(FilterConfig filterConfig) throws ServletException {
18
19 }
20
21 public void doFilter(ServletRequest req, ServletResponse res,
22 FilterChain chain) throws IOException, ServletException {
23 final HttpServletRequest request = (HttpServletRequest)req;
24 HttpServletResponse response = (HttpServletResponse)res;
25 request.setCharacterEncoding("utf-8");
26 response.setContentType("text/html;charset=utf-8");
27 //使用代理來擴充request對象的getParameter方法
28 HttpServletRequest _request = (HttpServletRequest)Proxy.newProxyInstance(request.getClass().getClassLoader()
29 , new Class[]{HttpServletRequest.class}, new InvocationHandler() {
30
31 public Object invoke(Object proxy, Method method, Object[] args)
32 throws Throwable {
33 String methodName = method.getName();
34 if("getParameter".equals(methodName)){
35 //request中的getParameter方法,當get送出時,需要對編碼進行處理
36 if("GET".equals(request.getMethod())){
37 String content = (String)method.invoke(request, args[0].toString());//request.getParameter(args[0].toString());
38 return new String(content.getBytes("ISO8859-1"),"utf-8");
39 }
40 }
41 return method.invoke(request, args);
42 }
43 });
44
45 chain.doFilter(_request, response);//放行
46 request.getRequestDispatcher("index.jsp").forward(request, response);
47 }
48
49 public void destroy() {
50
51 }
52
53
54 }
2、在web.xml中配置好過濾器的映射關系
1 <?xml version="1.0" encoding="UTF-8"?>
2 <web-app version="2.5"
3 xmlns="http://java.sun.com/xml/ns/javaee"
4 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
5 xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
6 http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
7 <!------------------------Filter映射配置部分------------------------------->
8 <filter>
9 <filter-name>LoginFilter</filter-name>
10 <filter-class>fnz.filter.webFilter</filter-class>
11 </filter>
12 <filter-mapping>
13 <filter-name>LoginFilter</filter-name>
14 <url-pattern>/*</url-pattern>
15 </filter-mapping>
16 <!------------------------Filter映射配置部分------------------------------->
17 <display-name></display-name>
18 <servlet>
19 <description>This is the description of my J2EE component</description>
20 <display-name>This is the display name of my J2EE component</display-name>
21 <servlet-name>LoginServlet</servlet-name>
22 <servlet-class>fnz.servlet.LoginServlet</servlet-class>
23 </servlet>
24
25 <servlet-mapping>
26 <servlet-name>LoginServlet</servlet-name>
27 <url-pattern>/LoginServlet</url-pattern>
28 </servlet-mapping>
29 <welcome-file-list>
30 <welcome-file>index.jsp</welcome-file>
31 </welcome-file-list>
32 </web-app>
3、最後編寫一個servlet(LoginServlet.java)及前台頁面(index.jsp),用于測試Filter
LoginServlet.java
1 package fnz.servlet;
2
3 import java.io.IOException;
4 import java.io.PrintWriter;
5
6 import javax.servlet.ServletException;
7 import javax.servlet.http.HttpServlet;
8 import javax.servlet.http.HttpServletRequest;
9 import javax.servlet.http.HttpServletResponse;
10
11 public class LoginServlet extends HttpServlet {
12
13 public void doGet(HttpServletRequest request, HttpServletResponse response)
14 throws ServletException, IOException {
15
16 System.out.println(request.getParameter("name"));
17 }
18
19 public void doPost(HttpServletRequest request, HttpServletResponse response)
20 throws ServletException, IOException {
21 this.doGet(request, response);
22 }
23 }
index.jsp
1 <%@ page language="java" import="java.util.*" pageEncoding="utf-8"%>
2
3 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
4 <html>
5 <head>
6 <title>My JSP 'index.jsp' starting page</title>
7 </head>
8 <body>
9 <form action="${pageContext.request.contextPath}/LoginServlet" method="post">
10 <input type="text" name="name"/>
11 <input type="submit" value="POST送出"/>
12 </form>
13 <form action="${pageContext.request.contextPath}/LoginServlet" method="get">
14 <input type="text" name="name"/>
15 <input type="submit" value="GET送出"/>
16 </form>
17 </body>
18 </html>
此案例目錄結構:
end