一、Servlet类图
Servlet接口定义了几个关键的方法:
- init(),初始化 servlet 对象,完成一些初始化工作。它是由 servlet 容器控制的,该方法只能被调用一次
- service(),接受客户端请求对象,执行业务操作,利用响应对象响应客户端请求。
- destroy(),当容器监测到一个servlet从服务中被移除时,容器调用该方法,释放资源,该方法只能被调用一次。
- getServletConfig(),ServletConfig 是容器向 servlet 传递参数的载体。
- getServletInfo(),获取 servlet 相关信息。
init() 方法
init 方法被设计成只调用一次。它在第一次创建 Servlet 时被调用,在后续每次用户请求时不再调用。
service() 方法
service() 方法是执行实际任务的主要方法。Servlet 容器(即 Web 服务器)调用 service() 方法来处理来自客户端(浏览器)的请求,并把格式化的响应写回给客户端。
每次服务器接收到一个 Servlet 请求时,服务器会产生一个新的线程并调用服务。service() 方法检查 HTTP 请求类型(GET、POST、PUT、DELETE 等),并在适当的时候调用 doGet、doPost、doPut,doDelete 等方法。
destroy() 方法
destroy() 方法只会被调用一次,在 Servlet 生命周期结束时被调用。destroy() 方法可以让您的 Servlet 关闭数据库连接、停止后台线程、把 Cookie 列表或点击计数器写入到磁盘,并执行其他类似的清理活动。
在调用 destroy() 方法之后,servlet 对象被标记为垃圾回收。
二、何时创建
一般,创建servlet时机有两个
1.第一次访问Servlet的时候创建
(1).类加载
(2).实例化一个servlet 创建Servlet对象
(3).init方法
做一些初始化工作..比如从web.xml文件中读取配置信息
(4).service方法
来处理客户端请求 判断请求类型get 调用doGet,post 调用doPost 在该方法中完成
(5).destroy方法
程序员能主动调用destroy方法来销毁servlet对象吗?
答:不能,这些方法都是tomcat容器调用的
destroy方法 什么时候会被调用
(1)web服务器关闭时
(2) web应用程序关闭时(或被移除时)
(3)如果一个Servlet长时间没有访问 调用destory方法
具体执行策略:会先把对象 保存到文件中(使用对象序列化技术) 等再访问该servlet对象时使用返序列化技术恢复Servlet
这样Servlet的状态就不会被丢失了
6.类卸载
重点:只有一个Servlet对象
对于同一个Servlet类的请求 只有一个Servlet对象
相当于每来一个新的请求 容器会为这个新的请求启动一个线程 并且在该线程内调用service()方法。
执行过程:
第一次访问时 先类加载、创建对象、调用init方法初始化、service方法
第二次再访问 前面的都不执行了,直接执行service方法
2. web服务器启动时创建:在web.xml servlet标签里配如下参数 <load-on-startup>1</load-on-startup>
源代码分析:
在StandardContex 里面有个addServletMappingDecoded 方法:
@Override public void addServletMappingDecoded(String pattern, String name, boolean jspWildCard) { // Validate the proposed mapping if (findChild(name) == null) throw new IllegalArgumentException (sm.getString("standardContext.servletMap.name", name)); String adjustedPattern = adjustURLPattern(pattern); if (!validateURLPattern(adjustedPattern)) throw new IllegalArgumentException (sm.getString("standardContext.servletMap.pattern", adjustedPattern)); // Add this mapping to our registered set synchronized (servletMappingsLock) { String name2 = servletMappings.get(adjustedPattern); if (name2 != null) { // Don't allow more than one servlet on the same pattern Wrapper wrapper = (Wrapper) findChild(name2); wrapper.removeMapping(adjustedPattern); } servletMappings.put(adjustedPattern, name); } Wrapper wrapper = (Wrapper) findChild(name); wrapper.addMapping(adjustedPattern); fireContainerEvent("addServletMapping", adjustedPattern); }
初始化的时候会往servletMappings 里面增加一个url,servlet 映射。 执行的时候从映射结合集合里面取出来servlet。
三、架构图
下图显示了一个典型的 Servlet 生命周期方案。
- 第一个到达服务器的 HTTP 请求被委派到 Servlet 容器。
- Servlet 容器在调用 service() 方法之前加载 Servlet。
- 然后 Servlet 容器处理由多个线程产生的多个请求,每个线程执行一个单一的 Servlet 实例的 service() 方法。