Servlet和filter是J2EE開發中常用的技術,使用方便,配置簡單,老少皆宜。
估計大多數朋友都是直接配置用,也沒有關心過具體的細節,今天遇到一個問題,上網查了Servlet的規范才發現,Servlet和filter中的url-pattern還是有一些文章在裡面的,總結了一些東西,放出來供大家參考,以免遇到問題又要浪費時間。
一、Servlet容器對url的匹配過程:
當一個請求發送到Servlet容器的時候,容器先會將請求的url減去當前應用上下文的路徑作為Servlet的映射url,比如我訪問的是http://localhost/test/aaa.html,我的應用上下文是test,容器會將http://localhost/test去掉,剩下的/aaa.html部分拿來做Servlet的映射匹配。這個映射匹配過程是有順序的,而且當有一個Servlet匹配成功以後,就不會去理會剩下的Servlet了filter不同,後文會提到)。其匹配規則和順序如下:
1. 精確路徑匹配。例子:比如ServletA 的url-pattern為 /test,ServletB的url-pattern為 /* ,這個時候,如果我訪問的url為http://localhost/test ,這個時候容器就會先進行精確路徑匹配,發現/test正好被ServletA精確匹配,那麼就去調用ServletA,也不會去理會其他的Servlet了。
2. 最長路徑匹配。例子:ServletA的url-pattern為/test/*,而ServletB的url-pattern為/test/a/*,此時訪問http://localhost/test/a時,容器會選擇路徑最長的Servlet來匹配,也就是這裡的ServletB.
3. 擴展匹配,如果url最後一段包含擴展,容器將會根據擴展選擇合適的Servlet.例子:ServletA的url-pattern:*.action
4. 如果前面三條規則都沒有找到一個Servlet,容器會根據url選擇對應的請求資源。如果應用定義了一個default Servlet,則容器會將請求丟給default Servlet什麼是default Servlet?後面會講)。
根據這個規則表,就能很清楚的知道Servlet的匹配過程,所以定義Servlet的時候也要考慮url-pattern的寫法,以免出錯。
對於filter,不會像Servlet那樣只匹配一個Servlet,因為filter的集合是一個鏈,所以只會有處理的順序不同,而不會出現只選擇一個filter.Filter的處理順序和filter-mapping在web.xml中定義的順序相同。
二、url-pattern詳解
在web.xml文件中,以下語法用於定義映射:
以“/‘開頭和以”/*“結尾的是用來做路徑映射的。
以前綴“*.”開頭的是用來做擴展映射的。
“是用來定義default Servlet映射的。
剩下的都是用來定義詳細映射的。比如: /aa/bb/cc.action
所以,為什麼定義“/*.action”這樣一個看起來很正常的匹配會錯?因為這個匹配即屬於路徑映射,也屬於擴展映射,導致Servlet容器無法判斷。