Web服务器如何确定哪个Servlet请求
文章目录
Servlet
Servlet是服务器端的Java应用程序,可以生产动态Web页面。透过JSP执行过程可以知道JSP最终被编译成一个.class文件,查看该文件对应的Java类,发现该Java类继承自org.apache.jasper.runtime.HttpJspBase类,而HttpJspBase继承自HttpServlet类,由此可知JSP第一次运行时实质上是被JSP引擎翻译成了一个Servlet,然后再编译,最后再执行。
自定义Servlet类继承HttpServlet抽象类,HttpServlet抽象类继承自GenericServlet抽象类,GenericServlet抽象类实现了Servlet、ServletConfig和Serializable接口。
Servlet生命周期
1. 加载及实例化
Servlet容器负责加载和实例化Servlet。当客户端第一次给服务器发送该Servlet请求时,Servlet容器会加载并创建Servlet实例。(注意:默认情况下不是Tomcat服务器或服务器上的Web应用启动的时候加载并实例化Servlet。在web.xml文件中,通过load-on-startup标签可以配置Servlet,当web项目发布后立即创建Servlet实例)当客户端(可以是非第一次请求的客户端)再次向服务器发送该Servlet请求时,服务器会从内存中查找该Servlet实例,并用找到的Servlet实例处理用户请求。
在该过程中,Servlet容器会创建一个ServletConfig对象,该对象包含了Servlet的初始化配置信息。根据用户请求的URL地址,Servlet容器会根据配置信息查找该请求对应的Servlet类,由容器创建并管理该Servlet。
2. 初始化——>init()
- 在Servlet容器完成Servlet类的实例化操作后,Servlet容器会调用Servlet的init()方法(在javax.servelt.Servlet接口中定义)对该Servlet进行初始化。
- 对于每一个Servlet实例来说,init()方法只会被调用一次。
- 初始化的目的是让Servlet在处理用户请求之前,做一些必要的准备工作,例如建立数据库连接,引用其他资源等。
3. 处理请求——>service()——>doGet()或doPost()
- Servlet初始化之后,就处于就绪状态等待接收用户请求。
- 当Servlet容器接收到客户端针对该Servlet的请求后,首先会针对这个请求创建ServletRequest和ServletResponse对象,之后调用Servlet的service()方法并把这两个参数传递给service()方法处理客户端请求。Servlet实例通过ServletRequest对象获得客户端的请求,通过调用ServletResponse对象的方法进行响应。
- 请求处理完毕,ServletRequest和ServletResponse对象被销毁。
- 不管客户端发送请求的方式是Get还是Post,这个请求都由service()方法来处理。在service方法的处理过程中,会根据客户端发送请求的方式不同,调用doGet和doPost方法分别进行处理,通过HttpServlet类中的service方法可以了解这一调用过程。
4. 销毁
- 销毁Servlet 由Servlet容器完成。
- 默认情况下,用户第一次发送Servlet请求,该Servlet加载、实例化、初始化、处理用户请求,当请求处理完毕后,该Servlet通常情况下驻留在内存中,等待处理下一个针对该Servlet的请求。当下一个针对该Servlet的请求到达时,直接从内存中获取该Servlet实例并对该请求进行处理。如果Tomcat这个Web应用服务器关闭(服务器上所有的Web应用都关闭),或者该Servlet所在的Web应用关闭,该Servlet实例会被销毁。
- Web应用被关闭时,Servlet容器会先调用Servlet实例的destroy方法,然后再销毁Servlet实例,同时也会销毁与Servlet相关联的ServletConfig对象。
- 程序员通常在destroy()方法的实现中释放该Servlet所占用的资源,如关闭数据库连接,关闭文件输入/输出流等。
Servlet生命周期(精简版)
- init()方法在servlet生命周期的初始化阶段被调用。它传递一个实现了javax.servlet.ServletConfig接口的对象,使得servlet能够从web application中获取初始化参数。
- servlet初始化收,每接收一个请求,就会调用service()方法。每个请求的处理都在独立的线程中进行。Web服务器对每个请求都会调用一次service()方法。service()方法判断请求的类型,并把它转发给相应的方法进行处理。
即: Servlet 容器处理由多个线程产生的多个请求,每个线程执行一个单一的 Servlet 实例的 service() 方法。 - 当需要销毁servlet对象时,就要调用destroy()方法。该方法释放被占用的资源。
Servlet2.X配置
1. 配置servlet元素和servlet-mapping元素
在web.xml文件中,通过在节点下配置servlet元素和servlet-mapping元素,把用户访问的URL映射到指定的Servlet类,如下代码:
<web-app>
<!-- 省略其他配置 -->
<servlet>
<!-- servlet-name指定Servlet名,要与下面servlet-mapping元素中的servlet-name保持一致 -->
<servlet-name>doLogin</servlet-name>
<!-- servlet-class对应着Servlet类完全限定名 -->
<servlet-class>com.jd.serlvet.LoginServlet</servlet-class>
</servlet>
<servlet-mapping>
<!-- servlet-name要与上面servlet元素中的servlet-name保持一致 -->
<servlet-name>doLogin</servlet-name>
<!-- url-pattern设定当前Servlet在浏览器中运行时的url -->
<url-pattern>/doLogin</url-pattern>
</servlet-mapping>
</web-app>
上面采用了精确匹配的形式配置了URL到Servlet之间的映射关系,接下来介绍两种非精确匹配的Servlet配置方式:
<!-- 对doLogin路径下的所有请求都由doLogin对应的Servlet类进行处理 -->
<servlet-mapping>
<servlet-name>doLogin</servlet-name>
<url-pattern>/doLogin/*</url-pattern>
</servlet-mapping>
<!-- 对所有以.do为后缀的请求都由doLogin对应的Servlet类进行处理 -->
<servlet-mapping>
<servlet-name>doLogin</servlet-name>
<!-- 不能为/*.do -->
<url-pattern>*.do</url-pattern>
</servlet-mapping>
在配置了URL与Servlet的映射后,当Servlet容器收到一个请求时,首先确定是由哪个Web应用响应这个请求,然后从该Web应用的web.xml文件中查找URL对应的Servlet类进行处理。
2. Servlet初始化参数设置
在web.xml文件中配置Servlet时,还可以在servlet元素中添加init-param元素预先对Servlet进行初始化设置,当Servlet加载时即可从该Servlet配置文件中获取初始化参数。
<!-- 省略其他配置 -->
<servlet>
<servlet-name>doLogin</servlet-name>
<servlet-class>com.jd.serlvet.LoginServlet</servlet-class>
<!-- 配置多个初始化参数,则需要写多个init-param元素 -->
<init-param>
<param-name>name</param-name>
<param-value>Tom</param-value>
</init-param>
</servlet>
<servlet-mapping>
<servlet-name>doLogin</servlet-name>
<url-pattern>/doLogin</url-pattern>
</servlet-mapping>
如何获取:
(1)在无参init方法中直接调用getInitParameter(String name)方法即可,如下代码:
@Override
public void init() throws ServletException {
String name = getInitParameter("name");
System.out.println(name);
}
(2)在参数为ServletConfig的方法中调用ServletConfig内getInitParameter(String name)方法即可,如下代码:
@Override
public void init(ServletConfig config) throws ServletException {
String name = config.getInitParameter("name");
System.out.println(name);
}
3. Servlet上下文(环境对象)初始化参数设置
有时候不仅需要针对单个Servlet进行初始化参数设置,还需要对包含该Web引用中所有Servlet的环境对象进行初始化参数设置,使该参数能被所有的Servlet共享,如下代码:
<!-- 省略其他配置 -->
<!-- 配置多个初始化参数,则需要写多个context-param元素 -->
<context-param>
<param-name>name</param-name>
<param-value>Tom</param-value>
</context-param>
<servlet>
<servlet-name>doLogin</servlet-name>
<servlet-class>com.jd.serlvet.LoginServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>doLogin</servlet-name>
<url-pattern>/doLogin</url-pattern>
</servlet-mapping>
</web-app>
如何获取:
(1)在无参init方法中直接调用getServletContext ()方法获取Servlet上下文对象,然后使用该对象调用getInitParameter方法即可,如下代码:
@Override
public void init() throws ServletException {
String name = getServletContext ().getInitParameter("name");
System.out.println(name);
}
(2)在参数为ServletConfig的方法中调用ServletConfig内getServletContext ()方法获取Servlet上下文对象,然后使用该对象调用getInitParameter方法即可,如下代码:
@Override
public void init(ServletConfig config) throws ServletException {
ServletContext servletContext = config.getServletContext();
String name = servletContext.getInitParameter("name");
System.out.println(name);
}
Web服务器如何确定哪个Servlet请求
- Web服务器的Web Server接收到HTTP请求
- Web服务器将请求转发给servlet容器
- servlet容器根据请求的URL去调用客户端要访问的Servlet
- 在调用之前,servlet容器会根据web.xml中对Servlet的描述去查找要访问的Servlet
-
若找到,将此Servlet装载进虚拟机并调用servlet的init()方法对servlet进行初始化(该方法只会在servlet第一次被载入时调用),然后调用Servlet实例中的service()方法处理请求,即,读取请求中的数据,创建一个响应。servlet会被保留在容器的地址空间中,继续处理其他的HTTP请求。
注: 当第二次去访问同一个Servlet时,若容器判断到该Servlet已经被装载并实例化,那么容器就不会再去创建一个新的Servlet实例,直接调用原来那个Servlet实例中的service()方法来处理请求。
-
若没有找到,直接返回一个404的错误代码到客户端,表示访问的资源不存在。
-
-
注:但如果在Spring MVC中的web.xml中添加如下代码,就可以交给Web服务器默认Servlet处理。
<mvc:default-servlet-handler/> <mvc:annotation-driven></mvc:annotation-driven>
- Web服务器将动态生成的结果返回到正确的地址。