Servlet虛擬路徑映射配置詳解
在上一篇中我們初識(shí)了Servlet,相信大家對(duì)Servlet也都有了些了解,知道了如何創(chuàng)建一個(gè)Servlet,并且為其添加虛擬映射,最終發(fā)布項(xiàng)目,并在瀏覽器上請(qǐng)求對(duì)應(yīng)的Servlet。
我們知道,只有給Servlet配置好虛擬路徑,客戶端才可以進(jìn)行訪問,但是對(duì)于Servlet的路徑映射,真的只有現(xiàn)在所知的這么簡(jiǎn)單么?
答案當(dāng)時(shí)是No了,不然怎么會(huì)有這篇文章😝,下面讓我們一起來探究其中的秘密吧!
Servlet虛擬路徑映射
在web.xml文件中,一個(gè)<servlet-mapping>元素用于映射一個(gè)Servlet的對(duì)外訪問路徑,該路徑也稱為虛擬路徑。例如<url-pattern>/TestServlet</url-pattern>,其中“/TestServlet”就是一個(gè)虛擬路徑。
1.配置多個(gè)映射路徑
在上一文中,我們說到@WebServlet中的urlPatterns屬性,其可以是一組匹配規(guī)則,也就是說一個(gè)Servlet是可以配置多個(gè)虛擬路徑的,也就是Servlet和虛擬路徑可以是一對(duì)多的一個(gè)關(guān)系(并不是多對(duì)多,一個(gè)虛擬路徑只能映射一個(gè)Servlet),其具體實(shí)現(xiàn)如下,并修改doPost處的代碼:
@WebServlet( description = "My First Servlet", urlPatterns = { "/HelloServlet", "/StillMe" }, initParams = { @WebInitParam(name = "name", value = "lizishu") }) public class HelloServlet extends HttpServlet { //具體邏輯參看上篇文章 //... protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //設(shè)置返回客戶端的contentType //text/plain :純文本格式 設(shè)置為text/html println的換行會(huì)失效 response.setContentType("text/plain;charset=utf-8"); //response.setCharacterEncoding("utf-8"); PrintWriter out = response.getWriter(); out.println("Served at: " + request.getContextPath()); String name = this.getInitParameter("name"); out.println("name: " + name); out.println("訪問的Servle名為:" + HelloServlet.class); } }
可以看到,增加一個(gè)虛擬路徑映射非常方便,只需在urlPatterns中新增一項(xiàng)即可(注意'/'不可省略),啟動(dòng)項(xiàng)目,在瀏覽器上輸入url,可以看到,無論是輸入http://localhost:8080/FirstProject/HelloServlet、還是http://localhost:8080/FirstProject/StillMe頁面上得到的輸出內(nèi)容均一致。
urlPatterns在Servlet 3.0版本之前,都是配置在web.xml中的,每個(gè)Servlet會(huì)有一個(gè)對(duì)應(yīng)的<servlet-mapping>
標(biāo)簽,其中可以配置多個(gè)<url-pattern>
。
2.urlPatterns匹配規(guī)則
說到Servlet虛擬路勁的匹配規(guī)則,還需要說到urlPatterns的幾種匹配規(guī)則,主要有以下四種:
- 精確匹配:也就是我們?cè)谏厦媾渲玫钠ヅ湟?guī)則,需要完全相等才能匹配成功,這也是我們經(jīng)常發(fā)生錯(cuò)誤的地方,請(qǐng)求Servlet時(shí)的大小寫拼寫錯(cuò)誤導(dǎo)致404;
- 路徑匹配:比如想匹配以rest開頭的所有請(qǐng)求,可以寫成"/rest/*",其格式為以'/‘字符開頭,并以'/*'結(jié)尾;
- 擴(kuò)展名匹配:比如想匹配所有以.do結(jié)尾的請(qǐng)求,可以寫成"*.do",其格式為以'*.',后面跟上擴(kuò)展名;
- 缺省匹配:映射路徑為"/",那么這個(gè)Servlet就是當(dāng)前應(yīng)用的缺省Servlet,默認(rèn)處理無法匹配到虛擬路徑的請(qǐng)求。
需要注意的是,路徑匹配和擴(kuò)展匹配無法混合使用,即urlPattern無法寫成"/rest/*.do";這也是讓部分同學(xué)感到困惑的地方,Servlet的虛擬路徑匹配并不是完全的按照正則來匹配的,雖然路徑匹配和擴(kuò)展匹配是按照正則中的通配符(*)來匹配的,這也是部分同學(xué)可以會(huì)寫出特定的正則,但是卻不是一個(gè)合法的虛擬路徑;Servlet容器收到請(qǐng)求后,會(huì)將請(qǐng)求從上下文路徑(通過request.getContextPath()獲取的)處截?cái)?,使用剩余的部分來進(jìn)行路徑匹配,比如請(qǐng)求url為http://localhost:8080/FirstProject/HelloServlet,那么Servlet容器就會(huì)使用"/HelloServlet"來匹配Servlet。
最后需要注意的是,我們說了上面四種匹配規(guī)則,尤其是缺省匹配,可以匹配到任意請(qǐng)求,那么一個(gè)請(qǐng)求如果可以匹配多個(gè)Servlet的虛擬路徑,那么該執(zhí)行哪個(gè)Servlet?其實(shí)啊,這些匹配規(guī)則是有優(yōu)先級(jí)的,具體的優(yōu)先級(jí)為:精確匹配>路徑匹配>擴(kuò)展名匹配>缺省匹配,Servlet容器會(huì)從優(yōu)先級(jí)高的虛擬路徑開始匹配,匹配到后就會(huì)立刻將請(qǐng)求交給對(duì)應(yīng)的Servlet來處理,不會(huì)再關(guān)心其他Servlet的虛擬路徑是否會(huì)匹配成功。
下面我們來一組Servlet及其對(duì)應(yīng)的虛擬路徑:
urlPatterns | Servlet Name |
---|---|
/abc/* | Servlet1 |
/ | Servlet2 |
/abc | Servlet3 |
*.do | Servlet4 |
當(dāng)請(qǐng)求去除上下文路徑后路徑為:"/abc/a.html"時(shí),根據(jù)上述規(guī)則,會(huì)調(diào)用Servlet1;
請(qǐng)求為:"/abc",根據(jù)匹配優(yōu)先級(jí),會(huì)調(diào)用Servlet3;
請(qǐng)求為:"/abc/a.do",會(huì)匹配到'/abc/*'、'*.do',但根據(jù)匹配優(yōu)先級(jí),會(huì)調(diào)用Servlet1;
請(qǐng)求為:"/a.do",會(huì)匹配到'/'、'*.do',但根據(jù)匹配優(yōu)先級(jí),會(huì)調(diào)用Servlet4;
3.Tomcat提供的缺省Servlet
為了測(cè)試缺省Servlet,我們來進(jìn)行一個(gè)測(cè)試。我們新建個(gè)SelfDefaultServlet,其urlPatterns我們配置為"/",其中的方法我們不做任何修改。
@WebServlet( description = "Self create default Servlet", urlPatterns = { "/" } ) public class SelfDefaultServlet extends HttpServlet { //... }
我們啟動(dòng)項(xiàng)目后,在瀏覽器上輸入http://localhost:8080/FirstProject/hahaha或者其他任意無法匹配到HelloServlet虛擬路徑的請(qǐng)求,發(fā)現(xiàn)頁面上的結(jié)果都如下所示,是不是這樣也不錯(cuò),不會(huì)報(bào)404錯(cuò)誤了。

但是,此時(shí)我們想訪問WebContent目錄下的靜態(tài)頁面(新建的一個(gè)welcome.html文件),瀏覽器上輸入http://localhost:8080/FirstProject/welcome.html,猜猜會(huì)發(fā)生什么?我們來一起看下結(jié)果,如圖所示,請(qǐng)求結(jié)果并沒有按照我們的想法,根據(jù)請(qǐng)求路徑找到welcome.htm頁面,而是調(diào)用了SelfDefaultServlet,是不是很懵?

其實(shí),客戶端的每個(gè)請(qǐng)求,都是由Servlet容器根據(jù)虛擬路徑的匹配規(guī)則來進(jìn)行處理的,包括靜態(tài)資源。并且,如果路徑輸入錯(cuò)誤(去除了自己配置的缺省Servlet后),我們常見的下面的錯(cuò)誤,也是Servlet返回給我們,哈哈,還是很意外?

我們能通過servlet方便簡(jiǎn)單的開發(fā)網(wǎng)站,是因?yàn)槲覀冋驹诹司奕说募绨蛏?,下面我們一起來看下Sun公司都為我們開發(fā)者提前做了些什么工作。Tomcat會(huì)為項(xiàng)目配置一個(gè)缺省的Servlet(如果項(xiàng)目中自行配置,則不會(huì)生效),配置文件在tomcat安裝目錄下conf目錄中的web.xml文件中,具體內(nèi)容如下,缺省的Servlet名為DefaultServlet。
<servlet> <servlet-name>default</servlet-name> <servlet-class>org.apache.catalina.servlets.DefaultServlet</servlet-class> <init-param> <param-name>debug</param-name> <param-value>0</param-value> </init-param> <init-param> <param-name>listings</param-name> <param-value>false</param-value> </init-param> <load-on-startup>1</load-on-startup> </servlet> <servlet-mapping> <servlet-name>default</servlet-name> <url-pattern>/</url-pattern> </servlet-mapping>
客戶端請(qǐng)求靜態(tài)資源文件時(shí),也是由缺省的Servlet處理的(自己?jiǎn)为?dú)配置Servlet除外),如果請(qǐng)求文件能找到,就會(huì)將頁面通過HttpServletResponse對(duì)象以流的方式返回給客戶端,否則報(bào)404錯(cuò)誤。
不過講到這里,大家可以自己試一試配置了缺省Servelt時(shí),訪問welcome.html的情況(會(huì)調(diào)用SelfDefaultServlet),但是,如果我們?cè)跒g覽器中輸入http://localhost:8080/FirstProject/index.jsp(index.jsp是創(chuàng)建的第一個(gè)jsp頁面)呢?會(huì)是什么樣一個(gè)結(jié)果?也是調(diào)用缺省的Servlet么?真是的運(yùn)行結(jié)果如下:

這是什么原因?為什么不是調(diào)用缺省的servlet了?這是因?yàn)閠omcat除了缺省Serlvet外,還給我們提供一個(gè)處理jsp文件的Servlet,配置如下,因?yàn)楹缶Y匹配的優(yōu)先級(jí)高于缺省的Servlet,所以訪問JSP的時(shí)候需要交由JspServlet來處理(JSP因?yàn)榭赡馨琂ava代碼,所以第一次執(zhí)行的時(shí)候需要先編譯,這個(gè)工作由JspServlet完成)
<servlet> <servlet-name>jsp</servlet-name> <servlet-class>org.apache.jasper.servlet.JspServlet</servlet-class> <init-param> <param-name>fork</param-name> <param-value>false</param-value> </init-param> <init-param> <param-name>xpoweredBy</param-name> <param-value>false</param-value> </init-param> <load-on-startup>3</load-on-startup> </servlet> <servlet-mapping> <servlet-name>jsp</servlet-name> <url-pattern>*.jsp</url-pattern> <url-pattern>*.jspx</url-pattern> </servlet-mapping>
4.總結(jié)
本文具體討論了urlPatterns屬性的匹配規(guī)則,主要為四種,其優(yōu)先級(jí)也各不相同,我們?cè)谑褂脮r(shí),也需要根據(jù)自己的需求自己設(shè)定urlPatterns,不過知道了匹配規(guī)則,使用起來也會(huì)方便很多,也能幫我們快速的定位錯(cuò)誤。
到此這篇關(guān)于Servlet虛擬路徑映射配置詳解的文章就介紹到這了,更多相關(guān)Servlet 虛擬路徑映射內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
mybatis中映射文件include標(biāo)簽的應(yīng)用
這篇文章主要介紹了mybatis中映射文件include標(biāo)簽的應(yīng)用,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11Springboot Cache @CacheEvict 無法模糊刪除的解決方案
這篇文章主要介紹了Springboot Cache @CacheEvict 無法模糊刪除的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-12-12Netty分布式NioEventLoop優(yōu)化selector源碼解析
這篇文章主要介紹了Netty分布式NioEventLoop優(yōu)化selector源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-03-03Netty啟動(dòng)流程服務(wù)端channel初始化源碼分析
這篇文章主要為大家介紹了Netty啟動(dòng)流程服務(wù)端channel初始化源碼分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-03-03基于Comparator對(duì)象集合實(shí)現(xiàn)多個(gè)條件按照優(yōu)先級(jí)的比較
這篇文章主要介紹了基于Comparator對(duì)象集合實(shí)現(xiàn)多個(gè)條件按照優(yōu)先級(jí)的比較,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07