Servlet部分细节介绍
1 Servlet与线程安全
因为一个类型的Servlet只有一个实例对象,那么就有可能会出现一个Servlet同时处理多个请求,那么Servlet是否为线程安全的呢?答案是:“不是线程安全的”。这说明Servlet的工作效率很高,但也存在线程安全问题!所以我们不应该在Servlet中随便创建成员变量,因为可能会存在一个线程对这个成员变量进行写操作,另一个线程对这个成员变量进行读操作。
2 让服务器在启动时就创建Servlet
默认情况下,服务器会在某个Servlet第一次收到请求时创建它。也可以在web.xml中对Servlet进行配置,使服务器启动时就创建Servlet。
1 <servlet> 2 <servlet-name>hello1</servlet-name> 3 <servlet-class>cn.servlet.Hello1Servlet</servlet-class> 4 <load-on-startup>0</load-on-startup>[在<servlet>中配置<load-on-startup>,其中给出一个非负整数!] 5 </servlet> 6 <servlet-mapping> 7 <servlet-name>hello1</servlet-name> 8 <url-pattern>/hello1</url-pattern> 9 </servlet-mapping> 10 <servlet> 11 <servlet-name>hello2</servlet-name> 12 <servlet-class>cn.servlet.Hello2Servlet</servlet-class> 13 <load-on-startup>1</load-on-startup> 14 </servlet> 15 <servlet-mapping> 16 <servlet-name>hello2</servlet-name> 17 <url-pattern>/hello2</url-pattern> 18 </servlet-mapping> 19 <servlet> 20 <servlet-name>hello3</servlet-name> 21 <servlet-class>cn.servlet.Hello3Servlet</servlet-class> 22 <load-on-startup>2</load-on-startup> 23 </servlet> 24 <servlet-mapping> 25 <servlet-name>hello3</servlet-name> 26 <url-pattern>/hello3</url-pattern> 27 </servlet-mapping>
在<servlet>元素中配置<load-on-startup>元素可以让服务器在启动时就创建该Servlet,其中<load-on-startup>元素的值必须是大于等于0的整数,它的使用是服务器启动时创建Servlet的顺序。上例中,根据<load-on-startup>的值可以得知服务器创建Servlet的顺序为Hello1Servlet、Hello2Servlet、Hello3Servlet。
3 <url-pattern>
<url-pattern>是<servlet-mapping>的子元素,用来指定Servlet的访问路径,即URL。它必须是以“/”开头!
1) 可以在<servlet-mapping>中给出多个<url-pattern>,例如:
1 <servlet-mapping> 2 <servlet-name>AServlet</servlet-name> 3 <url-pattern>/AServlet</url-pattern> 4 <url-pattern>/BServlet</url-pattern> 5 </servlet-mapping>
那么这说明一个Servlet绑定了两个URL,无论访问/AServlet还是/BServlet,访问的都是AServlet。
2) 还可以在<url-pattern>中使用通配符,所谓通配符就是星号“*”,星号可以匹配任何URL前缀或后缀,使用通配符可以命名一个Servlet绑定一组URL,例如:
l <url-pattern>/servlet/*<url-patter>:/servlet/a、/servlet/b,都匹配/servlet/*;
l <url-pattern>*.do</url-pattern>:/abc/def/ghi.do、/a.do,都匹配*.do;
l <url-pattern>/*<url-pattern>:匹配所有URL;
请注意,通配符要么为前缀,要么为后缀,不能出现在URL中间位置,也不能只有通配符。例如:/*.do就是错误的,因为星号出现在URL的中间位置上了。*.*也是不对的,因为一个URL中最多只能出现一个通配符。
注意,通配符是一种模糊匹配URL的方式,如果存在更具体的<url-pattern>,那么访问路径会去匹配具体的<url-pattern>。例如:
1 <servlet> 2 <servlet-name>hello1</servlet-name> 3 <servlet-class>cn.servlet.Hello1Servlet</servlet-class> 4 </servlet> 5 <servlet-mapping> 6 <servlet-name>hello1</servlet-name> 7 <url-pattern>/servlet/hello1</url-pattern> 8 </servlet-mapping> 9 <servlet> 10 <servlet-name>hello2</servlet-name> 11 <servlet-class>cn.servlet.Hello2Servlet</servlet-class> 12 </servlet> 13 <servlet-mapping> 14 <servlet-name>hello2</servlet-name> 15 <url-pattern>/servlet/*</url-pattern> 16 </servlet-mapping>
当访问路径为http://localhost:8080/hello/servlet/hello1时,因为访问路径即匹配hello1的<url-pattern>,又匹配hello2的<url-pattern>,但因为hello1的<url-pattern>中没有通配符,所以优先匹配,即访问hello1。
4 web.xml文件的继承(了解)
在${CATALINA_HOME}\conf\web.xml中的内容,相当于写到了每个项目的web.xml中,它是所有web.xml的父文件。每个完整的JavaWeb应用中都需要有web.xml,但我们不知道所有的web.xml文件都有一个共同的父文件,它在Tomcat的conf/web.xml路径。
conf/web.xml
<?xml version="1.0" encoding="ISO-8859-1"?> <web-app xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" version="3.0"> <servlet> <!-- 它的优先级最低,如果一个请求没有人处理,那么它来处理!它显示404 --> <servlet-name>default</servlet-name> <!-- 当访问路径不存在时,会执行该Servlet!其实我们在访问index.html时也是在执行这个Servlet --> <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> <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> <!-- 匹配所有URL,也就是说用户访问的URL路径没有匹配的页面时,那么执行的就是名为default的Servlet,即org.apache.catalina.servlets.DefaultServlet--> <servlet-mapping> <servlet-name>default</servlet-name> <url-pattern>/</url-pattern> </servlet-mapping> <!-- 任何URL后缀为jsp的访问,都会执行名为jsp的Servlet,即org.apache.jasper.servlet.JspServlet--> <servlet-mapping> <servlet-name>jsp</servlet-name> <url-pattern>*.jsp</url-pattern> <url-pattern>*.jspx</url-pattern> </servlet-mapping> <!-- session的默认超时时间为30分钟 --> <session-config> <session-timeout>30</session-timeout> </session-config> <!-- 这里省略了大概4000多行的MIME类型的定义,这里只给出两种MIME类型的定义 ,MIME类型用来标识网络上资源的媒体类型,这里举例为bmp和html两种MIME类型。--> <mime-mapping> <extension>bmp</extension> <mime-type>image/bmp</mime-type> </mime-mapping> <mime-mapping> <extension>htm</extension> <mime-type>text/html</mime-type> </mime-mapping> <!-- 在应用的web.xml中如果没有对<welcome-file-list>进行覆盖,那么默认主页为index.html、index.html、index.jsp--> <welcome-file-list> <welcome-file>index.html</welcome-file> <welcome-file>index.htm</welcome-file> <welcome-file>index.jsp</welcome-file> </welcome-file-list> </web-app>