web应用程序servlet的映射名称的规则及请求过程
首先用MyEclipse创建一个web Project(工程名起为TestServletProject),新建一个Servlet(这里servlet的名字起TestServlet),将请求的servlet映射名称设为/TestServlet,(具体步骤可以查看tomcat上servlet程序的配置与处理servlet请求过程)。并在TestServlet的doGet方法中在控制台打印一句“this is TestServlet”
jxf.servlet.TestServlet.java
1 package jxf.servlet;
2 import java.io.IOException;
3 import java.util.Date;
4 import java.io.PrintWriter;
5 import java.text.SimpleDateFormat;
6
7 import javax.servlet.ServletException;
8 import javax.servlet.http.HttpServlet;
9 import javax.servlet.http.HttpServletRequest;
10 import javax.servlet.http.HttpServletResponse;
11 public class TestServlet extends HttpServlet {
12 public void doGet(HttpServletRequest request, HttpServletResponse response)
13 throws ServletException, IOException {
14 response.getWriter().write(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date()));
15 System.out.println("this is TestServlet");
16 }
17 }
此时打开工程的web.xml配置文件
1 <?xml version="1.0" encoding="UTF-8"?>
2 <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" version="2.5">
3 <display-name></display-name>
4 <servlet>
5 <description>This is the description of my J2EE component</description>
6 <display-name>This is the display name of my J2EE component</display-name>
7 <!--3、 servlet的内部名称-->
8 <servlet-name>TestServlet</servlet-name>
9 <!--4、 servlet的类全名: 包名+简单类名 -->
10 <servlet-class>jxf.servlet.TestServlet</servlet-class>
11 </servlet>
12 <servlet-mapping>
13 <!--2、 servlet的映射内部名称,通过他可以找到上面的servlet的内部名称-->
14 <servlet-name>TestServlet</servlet-name>
15 <!--1、 请求servlet的映射路径-->
16 <url-pattern>/TestServlet</url-pattern>
17 </servlet-mapping>
18 <welcome-file-list>
19 <welcome-file>index.jsp</welcome-file>
20 </welcome-file-list>
21 </web-app>
将项目部署(在tomcat服务器)好后在浏览器中输入http://localhost:8081/TestServletProject/TestServlet(我的端口为8081),这时候服务器就会先找到TestServletProject工程下的web.xml,然后寻找这一次请求的映射路径/TestServlet,如上图绿色注释部分 根据映射/TestServlet找到注释1,然后找到同级servlet的映射内部名称注释2,在根据映射的内部名称找到servlet的内部名称注释3,最后找到同级的servlet的具体类名注释4,然后服务器在根据反射执行这个类的doGet方法...详细的Servlet请求的整个生命周期就不在这里讨论了。
最终在控制台输出this is TestServlet,并且页面上也显示出当前系统时间,说明该Servlet被请求并执行了。
本文主要介绍 请求servlet的映射路径 的写法:
上面的例子就是一种写法,为精确匹配
如:
/first http://localhost:8080/ProjectName/first
/xxx/demoServlet http://localhost:8080/ProjectName/xxx/demoServlet
还有一种写法是模糊匹配如:
/* http://localhost:8080/ProjectName/任意路径
/test/* http://localhost:8080/ProjectName/test/任意路径
*.后缀名 http://localhost:8080/ProjectName/任意路径.do 如:*.do *.action *.html(伪静态)
假如将上面的配置文件
<url-pattern>/TestServlet</url-pattern>改为<url-pattern>/*</url-pattern>
在浏览器地址栏中输入http://localhost:8080/ProjectName/xxxx(任意的请求映射路径/xxxx)一样可以请求到TestServlet.
注意:
1)url-pattern(请求servlet的映射路径)要么以 / 开头,要么以*开头。 例如, 只写test是非法路径。
2)不能同时使用两种模糊匹配,例如 /test/*.do是非法路径
3)当有输入的URL有多个servlet同时被匹配的情况下:
3.1 精确匹配优先。(长的最像优先被匹配,这里就做不验证了)
3.2 以后缀名结尾的模糊url-pattern优先级最低
4)虽然能够以/开头(/和/*两种写法是等价的),但是不推荐这种写法。为什么?
注意4解答:
当上面的配置文件<url-pattern>/TestServlet</url-pattern>改为<url-pattern>/*</url-pattern> 以后,我们想要访问项目中默认的静态文件index.html(创建项目后就自动有了),在浏览器中输入http://localhost:8080/ProjectName/index.html,发现请求的结果也是页面输出当前系统时间,控制台也输出this is TestServlet,可见其实最终访问到的是/TestServlet。因为 “请求servlet的映射路径”已经匹配到,服务器就当成是请求该servlet了。所以这里最好不要只写/或/*,因为项目中的其他静态资源都将无法访问到。
可以查看 tomcat根目录/conf/web.xml ,有下面一段代码
<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>
可以得知,/是servlet中预定义好的一个映射路径:servlet的缺省映射路径(<url-pattern>/</url-pattern>)是在tomcat服务器内置的一个映射路径。该路径对应的是一个DefaultServlet(缺省Servlet)。这个缺省的Servlet的作用是用于解析web应用的静态资源文件。
并且通过此我们还可以得出一个结论:先找动态资源,当动态资源不存在的时候,再找静态资源