Servlet,Filter的url-pattern詳解關(guān)鍵字: servlet filter url-pattern
Servlet和Filter的url匹配以及url-pattern詳解
Servlet和filter是J2EE開發(fā)中常用的技術(shù),使用方便,配置簡(jiǎn)單,老少皆宜。估計(jì)大多數(shù)朋友都是直接配置用,也沒(méi)有關(guān)心過(guò)具體的細(xì)節(jié),今天遇到一個(gè)問(wèn)題,上網(wǎng)查了servlet的規(guī)范才發(fā)現(xiàn),servlet和filter中的url-pattern還是有一些文章在里面的,總結(jié)了一些東西,放出來(lái)供大家參考,以免遇到問(wèn)題又要浪費(fèi)時(shí)間。
一,servlet容器對(duì)url的匹配過(guò)程:
當(dāng)一個(gè)請(qǐng)求發(fā)送到servlet容器的時(shí)候,容器先會(huì)將請(qǐng)求的url減去當(dāng)前應(yīng)用上下文的路徑作為servlet的映射url,比如我訪問(wèn)的是
http://localhost/test/aaa.html,我的應(yīng)用上下文是test,容器會(huì)將
http://localhost/test去掉,剩下的/aaa.html部分拿來(lái)做servlet的映射匹配。這個(gè)映射匹配過(guò)程是有順序的,而且當(dāng)有一個(gè)servlet匹配成功以后,就不會(huì)去理會(huì)剩下的servlet了(filter不同,后文會(huì)提到)。其匹配規(guī)則和順序如下:
1. 精確路徑匹配。例子:比如servletA 的url-pattern為 /test,servletB的url-pattern為 /* ,這個(gè)時(shí)候,如果我訪問(wèn)的url為
http://localhost/test ,這個(gè)時(shí)候容器就會(huì)先 進(jìn)行精確路徑匹配,發(fā)現(xiàn)/test正好被servletA精確匹配,那么就去調(diào)用servletA,也不會(huì)去理會(huì)其他的servlet了。
2. 最長(zhǎng)路徑匹配。例子:servletA的url-pattern為/test/*,而servletB的url-pattern為/test/a/*,此時(shí)訪問(wèn)
http://localhost/test/a時(shí),容器會(huì)選擇路徑最長(zhǎng)的servlet來(lái)匹配,也就是這里的servletB。
3. 擴(kuò)展匹配,如果url最后一段包含擴(kuò)展,容器將會(huì)根據(jù)擴(kuò)展選擇合適的servlet。例子:servletA的url-pattern:*.action
4. 如果前面三條規(guī)則都沒(méi)有找到一個(gè)servlet,容器會(huì)根據(jù)url選擇對(duì)應(yīng)的請(qǐng)求資源。如果應(yīng)用定義了一個(gè)default servlet,則容器會(huì)將請(qǐng)求丟給default servlet(什么是default servlet?后面會(huì)講)。
根據(jù)這個(gè)規(guī)則表,就能很清楚的知道servlet的匹配過(guò)程,所以定義servlet的時(shí)候也要考慮url-pattern的寫法,以免出錯(cuò)。
對(duì)于filter,不會(huì)像servlet那樣只匹配一個(gè)servlet,因?yàn)閒ilter的集合是一個(gè)鏈,所以只會(huì)有處理的順序不同,而不會(huì)出現(xiàn)只選擇一個(gè)filter。Filter的處理順序和filter-mapping在web.xml中定義的順序相同。
二,url-pattern詳解
在web.xml文件中,以下語(yǔ)法用于定義映射:
l 以”/’開頭和以”/*”結(jié)尾的是用來(lái)做路徑映射的。
l 以前綴”*.”開頭的是用來(lái)做擴(kuò)展映射的。
l “/” 是用來(lái)定義default servlet映射的。
l 剩下的都是用來(lái)定義詳細(xì)映射的。比如: /aa/bb/cc.action
所以,為什么定義”/*.action”這樣一個(gè)看起來(lái)很正常的匹配會(huì)錯(cuò)?因?yàn)檫@個(gè)匹配即屬于路徑映射,也屬于擴(kuò)展映射,導(dǎo)致容器無(wú)法判斷。