Java過(guò)濾器filter_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
Filter過(guò)濾器技術(shù)。通過(guò)過(guò)濾器,可以對(duì)來(lái)自客戶端的請(qǐng)求進(jìn)行攔截,進(jìn)行預(yù)處理或者對(duì)最終響應(yīng)給客戶端的數(shù)據(jù)進(jìn)行處理后再輸出。
要想使用Filter過(guò)濾器,非常簡(jiǎn)單,只要實(shí)現(xiàn)Servlet API中的Filter接口即可,同時(shí)在該web應(yīng)用【W(wǎng)EB-INF】目錄下的web.xml文件中配置<filter>和<filter-mapping>兩個(gè)標(biāo)簽。其中可以根據(jù)配置指定過(guò)濾的頁(yè)面或者Servlet。
也就是說(shuō)我們?cè)趙eb工程中光光寫Filter過(guò)濾器的Java代碼是不會(huì)起作用的,要在web.xml文件中對(duì)過(guò)濾器進(jìn)行注冊(cè)和映射,在學(xué)習(xí)Filter之前我們先來(lái)學(xué)習(xí)如何注冊(cè)和映射
關(guān)于注冊(cè):
需要在web.xml文件中配置<filter>標(biāo)簽,這還不夠,<filter>標(biāo)簽下的<filter-name>與<filter-class>是必須要填的內(nèi)容。
<filter>標(biāo)簽中有如下子元素:
- <description>用于描述該標(biāo)簽,非必須;
- <filter-name>為過(guò)濾器指定一個(gè)名稱,必須的
- <filter-class>指定該過(guò)濾器使用的web工程中的哪一個(gè)filter類,包含包名與類名,必須的;
- <init-param>為過(guò)濾器的初始化提供參數(shù),非必須,后面有例子。
關(guān)于映射:
需要在web.xml文件中配置<filter-mapping>標(biāo)簽,這還不夠,<filter-mapping>標(biāo)簽下的<filter-name>以及<url-pattern>或<servlet-name>之一是必須的。
<filter-mapping>標(biāo)簽中有如下子元素:
- <filter-name>設(shè)置要映射過(guò)濾器的名稱,該名稱必須同<filter>標(biāo)簽下的<filter-name>的值一致。
- <url-pattern>設(shè)置過(guò)濾器要攔截過(guò)濾的請(qǐng)求路徑,例如“/*”則表示對(duì)該web應(yīng)用下所有的請(qǐng)求都進(jìn)行攔截過(guò)濾。
- <servlet-name>如果只要攔截過(guò)濾訪問(wèn)某個(gè)Servlet,就可以使用該標(biāo)簽來(lái)替代<url-pattern>。
- <dispatcher>設(shè)置攔截過(guò)濾客戶端請(qǐng)求的方式,有REQUEST,INCLUDE,F(xiàn)ORWARD,ERROR四種(請(qǐng)注意均為大寫)。非必須則默認(rèn)為REQUEST,使用多個(gè)<dispatcher>標(biāo)簽來(lái)設(shè)置多種請(qǐng)求方式。
關(guān)于<dispathcer>的四種方式,這里再簡(jiǎn)單的介紹一下:
- REQUEST:當(dāng)用戶直接訪問(wèn)我們的資源時(shí),這時(shí)我們?cè)O(shè)置的過(guò)濾器就會(huì)進(jìn)行攔截。但如果以轉(zhuǎn)發(fā)和包含方式訪問(wèn)資源,那么該過(guò)濾器則不會(huì)被調(diào)用。
- INCLUDE:當(dāng)使用RequestDispatch的include方法請(qǐng)求時(shí),該過(guò)濾器會(huì)被調(diào)用。
- FORWARD:當(dāng)使用RequestDispatch的forward方法時(shí)請(qǐng)求資源時(shí),該過(guò)濾器會(huì)被調(diào)用,尤其是在MVC設(shè)計(jì)模式下,JSP都被保護(hù)起來(lái),必須要通過(guò)Servlet進(jìn)行轉(zhuǎn)發(fā)才能訪問(wèn)JSP,那么該過(guò)濾器就是在Servlet轉(zhuǎn)發(fā)到JSP這個(gè)過(guò)程中被執(zhí)行。
- ERROR:當(dāng)請(qǐng)求是通過(guò)錯(cuò)誤異常進(jìn)行跳轉(zhuǎn)時(shí)就會(huì)調(diào)用該過(guò)濾器。
一個(gè)簡(jiǎn)單的對(duì)過(guò)濾器的注冊(cè)和映射的示例:
<filter> <filter-name>FilterDemo1</filter-name> <filter-class>com.bjpowernode.web.filter.FilterDemo1</filter-class> </filter> <filter-mapping> <filter-name>FilterDemo1</filter-name> <url-pattern>/*</url-pattern> <dispatcher>REQUEST</dispatcher> <dispatcher>FORWARD</dispatcher> </filter-mapping>
在Servlet API 中關(guān)于Filter舉例了使用過(guò)濾器能用來(lái)做些什么:

這里我也說(shuō)明下平時(shí)Filter能在哪些方面會(huì)被經(jīng)常用到:
① Filter可以作用在請(qǐng)求資源執(zhí)行之前,進(jìn)行權(quán)限檢查,檢查用戶是否有權(quán)限,如有權(quán)限則放行請(qǐng)求;如果沒(méi)有,則拒絕訪問(wèn)。
② Filter可以作用在請(qǐng)求資源執(zhí)行之前,對(duì)Request和Response對(duì)象進(jìn)行預(yù)處理操作,從而實(shí)現(xiàn)一些web應(yīng)用的全局性設(shè)置,比如解決中文亂碼問(wèn)題。
③ Filter可以作用在最終響應(yīng)輸出之前,對(duì)輸出Response對(duì)象中的數(shù)據(jù)進(jìn)行處理,例如將輸出的數(shù)據(jù)進(jìn)行壓縮。
Filter只有3個(gè)方法:

其中destroy()方法和init(…)方法是生命周期方法,因?yàn)檫^(guò)濾器無(wú)論如何都要在請(qǐng)求任何資源之前進(jìn)行,所以任何Web應(yīng)用在部署的時(shí)候,服務(wù)器就會(huì)調(diào)用Filter過(guò)濾器的init方法進(jìn)行初始化,而關(guān)于過(guò)濾器的銷毀,則是將該過(guò)濾器移除或者服務(wù)器關(guān)閉就會(huì)執(zhí)行destory方法。
而我們通常要使用過(guò)濾器處理請(qǐng)求,則重點(diǎn)在于doFilter(…)方法。當(dāng)請(qǐng)求要經(jīng)過(guò)一個(gè)過(guò)濾器的時(shí)候,就會(huì)由服務(wù)器調(diào)用doFilter方法。
我們先來(lái)看看一個(gè)帶有過(guò)濾器Filter的web應(yīng)用的請(qǐng)求和響應(yīng)流程:

記?。簭恼?qǐng)求到響應(yīng)這個(gè)流程會(huì)經(jīng)過(guò)Filter對(duì)象兩次!
在doFilter這一個(gè)方法中就可以對(duì)著兩次經(jīng)過(guò)的過(guò)程進(jìn)行處理,那么這里就有一個(gè)問(wèn)題了,如果能通過(guò)過(guò)濾器,那么就到過(guò)濾器后面了,貌似應(yīng)該是執(zhí)行完doFilter方法了,而服務(wù)器的響應(yīng)又經(jīng)過(guò)過(guò)濾器,難道又要執(zhí)行doFilter方法一次?但是這個(gè)方法里面的代碼不是也有處理最開始請(qǐng)求的嗎?
這就跟doFilter方法中的第三個(gè)參數(shù)FilterChain有關(guān)了,F(xiàn)ilterChain對(duì)象是過(guò)濾器鏈,這個(gè)我們稍后會(huì)介紹。在FilterChain對(duì)象中只有一個(gè)方法:

=也是叫doFilter方法(千萬(wàn)別和Filter接口的doFilter方法弄混了)。簡(jiǎn)單的說(shuō)下這個(gè)方法,只要調(diào)用了這個(gè)方法,就會(huì)將請(qǐng)求交給后面一個(gè)Filter進(jìn)行過(guò)濾(一個(gè)Web應(yīng)用中可以有多個(gè)Filter),如果該Filter是最后一個(gè),那么調(diào)用該方法則將執(zhí)行請(qǐng)求,也就是到我們的應(yīng)用中獲取資源。
因此從請(qǐng)求到響應(yīng)這個(gè)流程經(jīng)過(guò)Filter的兩次處理分別是在FilterChain.doFilter方法的前面和后面!如下圖所示:

那么下面我們就先以一個(gè)簡(jiǎn)單的例子來(lái)熟悉下Filter吧:
例1:
創(chuàng)建web工程FilterLearning,創(chuàng)建一個(gè)FilterDemo1類,同時(shí)這個(gè)類要實(shí)現(xiàn)javax.servlet.Filter接口。如下代碼:
public class FilterDemo implements Filter {
@Override
public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain) throws IOException, ServletException {
System.out.println("hello filter");
}
//此處省略init方法和destory方法
}
寫好Filter的Java代碼還沒(méi)完,還要在web應(yīng)用下的web.xml文件中配置如下信息:
<filter>
<filter-name>FilterDemo1</filter-name>
<filter-class>com.bjpowernode.web.filter.FilterDemo</filter-class>
</filter>
<filter-mapping>
<filter-name>FilterDemo1</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
注意:因?yàn)槲以?lt;url-pattern>中配置為“/*”,則訪問(wèn)我wen應(yīng)用中任何資源都會(huì)經(jīng)過(guò)該Filter過(guò)濾器。如果只想對(duì)于index.jsp主頁(yè)的請(qǐng)求進(jìn)行過(guò)濾,可以設(shè)為<url-pattern>/index.jsp</url-pattern>。
我們?cè)趇ndex.jsp中簡(jiǎn)單的使用JSP腳本來(lái)演示如果有請(qǐng)求來(lái)就輸入一段文本到控制臺(tái)上:
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<!DOCTYPE HTML>
<html>
<head>
<title>index</title>
</head>
<body>
<%
System.out.println("Long live SD !");
%>
</body>
</html>
接下來(lái)將該web應(yīng)用部署到服務(wù)器中,我們就訪問(wèn)index.jsp,以下是我們?cè)L問(wèn)了index.jsp后控制臺(tái)的情況:

首先,我們可以保證在我們?cè)L問(wèn)index.jsp后這個(gè)請(qǐng)求確實(shí)經(jīng)過(guò)了Filter過(guò)濾器,但是我們的請(qǐng)求好像就只到過(guò)濾器而沒(méi)有到我們真正需要的資源index.jsp?這是因?yàn)槲覀儧](méi)有在Filter的doFilter方法中調(diào)用過(guò)濾器鏈FilterChain對(duì)象的doFilter方法,自然無(wú)法將請(qǐng)求繼續(xù)往后面?zhèn)鬟f。我們將在例2中修改。
例2:
我們將例1中的FilterDemo1類進(jìn)行修改,使其能訪問(wèn)到我們所需要的資源,很簡(jiǎn)單,在doFilter的方法中添加過(guò)濾器鏈FilterChain對(duì)象的doFilter方法即可:
public class FilterDemo implements Filter {
@Override
public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain) throws IOException, ServletException {
System.out.println("hello filter");
chain.doFilter(request, response);
}
//此處省略init方法和destory方法
}
其他如web.xml中的配置和index.jsp中的代碼保持不變,現(xiàn)在我們?cè)賮?lái)訪問(wèn)下該web應(yīng)用中的index.jsp,并觀察控制臺(tái):

可以看到我們的請(qǐng)求經(jīng)過(guò)過(guò)濾器,執(zhí)行了過(guò)濾器的一段代碼(System.out.println(“hello filter”)),然后將請(qǐng)求繼續(xù)執(zhí)行!正是因?yàn)镕ilterChain.doFilter方法才使我們通過(guò)過(guò)濾器繼續(xù)向后尋找我們所需的資源。
那么還記得我們之前說(shuō)過(guò)的從請(qǐng)求到響應(yīng)會(huì)經(jīng)過(guò)兩次過(guò)濾器嗎,是的在獲取了我們所需的資源后還會(huì)到過(guò)濾器一趟,而至于這時(shí)候是否將響應(yīng)再做處理取決于過(guò)濾器鏈FilterChain.doFilter方法后面還是否有代碼。我們將在例3中完整的展現(xiàn)從請(qǐng)求到響應(yīng)經(jīng)過(guò)過(guò)濾器兩次的流程。
例3:
我們將例2中的FilterDemo1類進(jìn)行修改,只要在FilterChain.doFilter方法后面添加代碼,就是第二次(即響應(yīng))經(jīng)過(guò)過(guò)濾器所要執(zhí)行的處理:
public class FilterDemo implements Filter {
@Override
public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain) throws IOException, ServletException {
System.out.println("hello filter");
chain.doFilter(request, response);
System.out.println("goodbye filter");
}
//此處省略init方法和destory方法
}
其他如web.xml中的配置和index.jsp中的代碼保持不變,現(xiàn)在我們?cè)賮?lái)訪問(wèn)下該web應(yīng)用中的index.jsp,并觀察控制臺(tái):

這個(gè)結(jié)果證明了從請(qǐng)求到響應(yīng)確實(shí)經(jīng)過(guò)過(guò)濾器兩次,同時(shí)也說(shuō)明了在Filter的doFilter方法中“過(guò)濾——取資源——再過(guò)濾”執(zhí)行的順序。
現(xiàn)在我們?cè)倩氐紽ilter接口的init方法,我們可以看到在這個(gè)方法內(nèi)有一個(gè)參數(shù)FilterConfig,這個(gè)是由服務(wù)器傳給我們的對(duì)象。如果我們?cè)趙eb.xml文件中配置了過(guò)濾器的初始化參數(shù),就可以通過(guò)該FilterConfig對(duì)象來(lái)在代碼中獲取使用。
這個(gè)過(guò)濾器參數(shù)的初始化配置可以在<filter>標(biāo)簽中配置<init-param>,并在這個(gè)<init-param>標(biāo)簽下再配置<param-name>和<param-value>。
FilterConfig有如下方法:

當(dāng)然如果我們是要獲取配置的初始化參數(shù)則只需關(guān)注getInitParameter方法或getInitParameterNames方法。
一般來(lái)說(shuō)我們可以在init方法中獲取配置初始化參數(shù)并進(jìn)行處理;也可以通過(guò)對(duì)象引用將FilterConfig對(duì)象在doFilter方法中處理參數(shù),如例4所示。
例4:
在web.xml文件中配置過(guò)濾器和初始化參數(shù):
<filter>
<filter-name>FilterDemo1</filter-name>
<filter-class>com.bjpowernode.web.filter.FilterDemo1</filter-class>
<init-param>
<param-name>Love</param-name>
<param-value>LRR</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>FilterDemo1</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
在Java中編寫Filter接口的實(shí)現(xiàn)類FilterDemo1:
public class FilterDemo implements Filter {
private FilterConfig filterConfig ;
@Override
public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain) throws IOException, ServletException {
String value = filterConfig.getInitParameter("Love");
System.out.println(value);
}
@Override
public void init(FilterConfig filterConfig) throws ServletException {
this.filterConfig = filterConfig;
}
//此處省略destroy方法
}
因?yàn)榕渲玫脑?,所以我們隨便訪問(wèn)個(gè)資源都可以經(jīng)過(guò)該過(guò)濾器,那么就訪問(wèn)index.jsp好了,看看控制臺(tái)的結(jié)果:

正如我們?cè)趙eb.xml文件所配置的初始化參數(shù)一樣。
上面介紹的都是只有一個(gè)Filter過(guò)濾器的情況下,有時(shí)候我們會(huì)因?yàn)橐^(guò)濾的功能不同添加多個(gè)過(guò)濾器,這就有一個(gè)順序的問(wèn)題了,尤其是從取得資源后再返回到過(guò)濾器的順序。下面這張圖就能很清晰的看到我們要注意的順序了:

例5:
來(lái)寫兩個(gè)Filter來(lái)說(shuō)明下從請(qǐng)求到響應(yīng)過(guò)濾器的處理順序。
創(chuàng)建一個(gè)web工程,創(chuàng)建一個(gè)FilterDemo1類,同時(shí)這個(gè)類要實(shí)現(xiàn)javax.servlet.Filter接口。如下代碼:
public class FilterDemo1 implements Filter {
public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain) throws IOException, ServletException {
System.out.println("Hello filter 1");
chain.doFilter(request, response);
System.out.println("Goodbye filter 1");
}
//此處省略init方法和destroy方法
}
創(chuàng)建第二個(gè)Filter接口實(shí)現(xiàn)類FilterDemo2,代碼如下:
public class FilterDemo2 implements Filter {
public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain) throws IOException, ServletException {
System.out.println("Hello filter 2");
chain.doFilter(request, response);
System.out.println("Goodbye filter 2");
}
//此處省略init方法和destroy方法
}
過(guò)濾器要想能被服務(wù)器調(diào)用,還必須要在該web工程下的web.xml中配置過(guò)濾器及其映射,而這個(gè)配置的順序就是影響多個(gè)過(guò)濾器工作先后的順序:
<filter>
<filter-name>FilterDemo1</filter-name>
<filter-class>com.bjpowernode.web.filter.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>com.bjpowernode.web.filter.FilterDemo2</filter-class>
</filter>
<filter-mapping>
<filter-name>FilterDemo2</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
而我們要訪問(wèn)的資源文件就以index.jsp為例好了,那么我們用一段JSP腳本通過(guò)在控制臺(tái)打印來(lái)驗(yàn)證過(guò)濾器工作的順序過(guò)程:
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<!DOCTYPE HTML>
<html>
<head>
<title>index</title>
</head>
<body>
<%
System.out.println("Long live SD !");
%>
</body>
</html>
現(xiàn)在啟動(dòng)服務(wù)器,部署該工程,通過(guò)訪問(wèn)index.jsp來(lái)看看控制臺(tái)情況:

相關(guān)文章
MyBatis中關(guān)于resultType和resultMap的區(qū)別介紹
MyBatis中在查詢進(jìn)行select映射的時(shí)候,返回類型可以用resultType,也可以用resultMap,那么MyBatis中關(guān)于resultType和resultMap的區(qū)別是什么呢?下面小編通過(guò)本文給大家解答下2016-09-09
如何解決idea安裝插件后報(bào)錯(cuò)打不開問(wèn)題
這篇文章主要介紹了如何解決idea安裝插件后報(bào)錯(cuò)打不開問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-06-06
java設(shè)計(jì)模式學(xué)習(xí)之工廠方法模式
這篇文章主要介紹了java設(shè)計(jì)模式學(xué)習(xí)之工廠方法模式,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-10-10
Java的作業(yè)調(diào)度類庫(kù)Quartz基本使用指南
這篇文章主要介紹了Java的作業(yè)調(diào)度類庫(kù)Quartz基本使用指南,Quartz能夠讓類按照指定的計(jì)劃順序執(zhí)行,需要的朋友可以參考下2016-03-03
java對(duì)象轉(zhuǎn)換String類型的三種方法
在很多情況下我們都需要將一個(gè)對(duì)象轉(zhuǎn)換為String類型。一般來(lái)說(shuō)有三種方法可以實(shí)現(xiàn):Object.toString()、(String)Object、String.valueOf(Object)。下面對(duì)這三種方法一一分析2013-11-11

