一、servlet容器對url的匹配過程: 當一個請求發送到servlet容器的時候,容器先會將請求的url減去當前應用上下文的路徑作為servlet的映射url,比如我訪問的是http://localhost/test/aaa.html(我的應用上下文是test),容器會將http://localhost/tes去掉,將剩下的/aaa.html部分拿來做servlet的映射匹配,也就是拿這剩下的部分與web.xml中配置的servlet的url-pattern進行匹配。注意:這個映射匹配過程是有一定的規則的,而且每次匹配最終都只匹配一個 servlet。(這一點和filter不同) 匹配規則如下:(它的匹配原則就是:找到唯一一個最適合的Servlet) 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(/test/*.action為不合法的url-pattern) 4. 如果前面三條規則都沒有找到一個servlet,容器會根據url選擇對應的請求資源。如果應用定義了一個default servlet,則容器會將請求丟給default servlet(什么是default servlet?后面會講)。 對於filter,不會像servlet那樣只匹配一個servlet,因為filter的集合是一個鏈,所以只會有處理的順序不同,而不會出現只選擇一個filter。Filter的處理順序和filter-mapping在web.xml中定義的順序相同。 二、url-pattern詳解 在web.xml文件中,以下語法用於定義映射: ① 以”/’開頭和以”/*”結尾的是用來做路徑映射的。 ② 以前綴”*.”開頭的是用來做擴展映射的。 ③ “/” 是用來定義default servlet映射的。 ④ 剩下的都是用來定義詳細映射的。比如: /aa/bb/cc.action 所以,為什么定義”/*.action”這樣一個看起來很正常的匹配會錯?因為這個匹配即屬於路徑映射,也屬於擴展映射,導致容器無法判斷。
一般做全匹配時,servlet的url-pattern 為 / 。filter的url-pattern 為 /* 。
此文章轉載自:http://www.cnblogs.com/kevin-yuan/archive/2012/12/24/2831333.html