JavaWeb学习-Servlet系列-3-Servlet的生命周期

上一篇介绍了Servlet的运行过程,知道了一个Servlet类如何在tomcat中被执行。这篇来学习Servlet的生命周期相关的话题。一个Servlet对象被创建之后,到被销毁之前,会执行哪些方法,这些问题是本篇学习需要掌握的。

1.阅读servlet j2ee api文档

注意这个已经不是JDK 1.6 chm那个文档了,Servlet接口只有在j2ee api chm这个文档才能查询得到。如果你本地没有这个文档,你可以去网上下载一个,搜索条件就输入j2ee api chm 中文版之类就能下载到本地。打开这个chm格式文件,搜索servet,内容如下:

这张图信息比较多,上一篇我们自定义一个servlet类实现了接口Servlet,然后需要我们实现这个接口内的全部方法,我们只在service这个方法中添加一个打印语句,其他方法都没管。

 

2.Servlet接口必须实现的方法

在前面Servlet初体验这篇文章,以下这几个方法就是实现Servlet接口必须要跟着实现的方法。

        public void service(ServletRequest req, ServletResponse res) throws ServletException, IOException {
		System.out.println("Hello Servlet.");
		
	}
	
	@Override
	public void destroy() {
		// TODO Auto-generated method stub
		
	}
 
	@Override
	public ServletConfig getServletConfig() {
		// TODO Auto-generated method stub
		return null;
	}
 
	@Override
	public String getServletInfo() {
		// TODO Auto-generated method stub
		return null;
	}
 
	@Override
	public void init(ServletConfig config) throws ServletException {
		// TODO Auto-generated method stub
		
	}

第一个方法是service(),服务的意思,第二个destroy()是销毁的意思,第三个getServletConfig(),得到Servlet配置信息的方法,第四个getServletInfo()得到Servlet作者信息的方法,第五个init()方法,初始化的意思。这五个方法都是Servlet接口的方法。

3.生命周期方法

在Servlet接口介绍文字中提到生存周期方法,而且按照一定顺序执行

对着API文档,初始化servlet其实说的就是Init()方法,服务于请求说的就是service()方法,从服务器删除说的是destroy()方法。然后我们来看上面1,2,3这顺序的意思。

先是执行Servlet自定义类的构造方法,默认是一个无参的空构造。然后执行init()进行初始化。第三service()方法,最后如果服务器把这个servlet删除,会执行destroy方法,最后垃圾回收是JVM干的活。

也就是生命周期三个方法,getServletConfig()和getServletInfo()两个方法不是重点,本篇不关注这两个方法,本篇重点来学习生命周期提到三个方法。init() service() destrou()

4.验证生命周期方法执行顺序

下面我们来验证下这个API文档说的生命周期中方法的执行顺序问题。

我们先准备一个web项目,和servlet初体验差不多一样。这里ServletDemo1.java代码贴出来,web.xml和前面这一篇是一样。

package com.anthony.servlet;

import java.io.IOException;

import javax.servlet.Servlet;
import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;

public class ServletDemo1 implements Servlet {
	// 把无参构造写出来,写一个打印语句,方便测试观察执行顺序
	public ServletDemo1() {
		System.out.println("********构造方法被执行*************");
	}
	
	@Override
	public void init(ServletConfig arg0) throws ServletException {
		System.out.println("********init方法被执行*************");
		
	}

	@Override
	public void service(ServletRequest arg0, ServletResponse arg1) throws ServletException, IOException {
		System.out.println("********service方法被执行*************");
		
	}
	
	@Override
	public void destroy() {
		System.out.println("********destroy方法被执行*************");
		
	}

	@Override
	public ServletConfig getServletConfig() {
		// TODO Auto-generated method stub
		return null;
	}

	@Override
	public String getServletInfo() {
		// TODO Auto-generated method stub
		return null;
	}

}

我分别在构造方法,init方法,service()和destroy()方法添加一行打印语句。把代码部署到tomcat上,然后点击启动tomcat服务,浏览器测试打开http://localhost:8080/Servlet01/demo

我这边可以在Eclipse上控制台看到这个输出

五月 06, 2019 10:30:36 下午 org.apache.catalina.startup.Catalina start
信息: Server startup in 914 ms
********构造方法被执行*************
********init方法被执行*************
********service方法被执行*************

上面的日志是我只请求了一次http://localhost:8080/Servlet01/demo的效果,方法执行效果,确实是先构造方法,然后init方法,然后service方法。下面我在浏览器多刷新几次http://localhost:8080/Servlet01/demo这个地址,再次看看Eclipse控制台输出

********构造方法被执行*************
********init方法被执行*************
********service方法被执行*************
********service方法被执行*************
********service方法被执行*************
********service方法被执行*************

得到一个结论,刷新几次,也就是servlet请求了几次,就会执行几次service()方法,而构造方法和init初始化方法只运行一次、下面就缺一个destroy方法被执行了,我这里Eclipse上直接点击关闭tomcat服务,也就是应用Servlet01服务器肯定挂了,看看能不能执行destroy方法。

信息: Server startup in 914 ms
********构造方法被执行*************
********init方法被执行*************
********service方法被执行*************
********service方法被执行*************
********service方法被执行*************
********service方法被执行*************
五月 06, 2019 10:37:26 下午 org.apache.catalina.core.StandardServer await
信息: A valid shutdown command was received via the shutdown port. Stopping the Server instance.
五月 06, 2019 10:37:26 下午 org.apache.coyote.AbstractProtocol pause
信息: Pausing ProtocolHandler ["http-nio-8080"]
五月 06, 2019 10:37:27 下午 org.apache.coyote.AbstractProtocol pause
信息: Pausing ProtocolHandler ["ajp-nio-8009"]
五月 06, 2019 10:37:27 下午 org.apache.catalina.core.StandardService stopInternal
信息: Stopping service [Catalina]
********destroy方法被执行*************
五月 06, 2019 10:37:27 下午 org.apache.catalina.core.ApplicationContext log
信息: SessionListener: contextDestroyed()
五月 06, 2019 10:37:27 下午 org.apache.catalina.core.ApplicationContext log
信息: ContextListener: contextDestroyed()
五月 06, 2019 10:37:27 下午 org.apache.coyote.AbstractProtocol stop
信息: Stopping ProtocolHandler ["http-nio-8080"]
五月 06, 2019 10:37:27 下午 org.apache.coyote.AbstractProtocol stop
信息: Stopping ProtocolHandler ["ajp-nio-8009"]
五月 06, 2019 10:37:27 下午 org.apache.coyote.AbstractProtocol destroy
信息: Destroying ProtocolHandler ["http-nio-8080"]
五月 06, 2019 10:37:27 下午 org.apache.coyote.AbstractProtocol destroy
信息: Destroying ProtocolHandler ["ajp-nio-8009"]

果然关闭了tomcat服务,这个servlet方法destroy()会被执行,也就是servlet这个对象就销毁了,直接让JVM垃圾回收了。上面这个过程就是一个servlet的生命周期过程,先执行构造方法,然后执行init方法初始化对象,然后浏览器发送几次请求,就执行几次service()方法,销毁方法只有在tomcat服务挂了或者应用项目Servlet01出现问题了才会被执行销毁,从而垃圾回收,这样完整的servlet生命周期过程就介绍完了。

5.小知识点-如何让servlet在tomcat服务器启动就创建呢

从上面我们知道,启动tomcat服务器之后,我们需要手动去浏览器触发请求,这个servlet才能被调用构造方法,然后调用init方法进行初始化。那么有没有办法,让tomcat服务器一起动就创建servlet对象并初始化。

这个需要在web.xml中添加一个标签,下面load-on-startup中2是权重的意思,数字越小,优先级越高,如果有多个servlet对象。

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://xmlns.jcp.org/xml/ns/javaee" xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd" id="WebApp_ID" version="3.1">
  
  <servlet>
  	<servlet-name>servletDemo1</servlet-name>
  	<servlet-class>com.anthony.servlet.ServletDemo1</servlet-class>
  	<load-on-startup>2</load-on-startup>
  </servlet>
  
  <servlet-mapping>
  	<servlet-name>servletDemo1</servlet-name>
  	<url-pattern>/demo</url-pattern>
  </servlet-mapping>
  
  <welcome-file-list>
    <welcome-file>index.html</welcome-file>
    <welcome-file>index.htm</welcome-file>
    <welcome-file>index.jsp</welcome-file>
    <welcome-file>default.html</welcome-file>
    <welcome-file>default.htm</welcome-file>
    <welcome-file>default.jsp</welcome-file>
  </welcome-file-list>
</web-app>

重新发布到tomcat,并启动tomcat看看Eclipse控制台输出效果。

信息: Starting service [Catalina]
五月 06, 2019 10:49:38 下午 org.apache.catalina.core.StandardEngine startInternal
信息: Starting Servlet Engine: Apache Tomcat/8.5.40
********构造方法被执行*************
********init方法被执行*************
五月 06, 2019 10:49:38 下午 org.apache.catalina.startup.HostConfig deployDirectory
信息: Deploying web application directory [F:\apache-tomcat-8.5.40\webapps\docs]
五月 06, 2019 10:49:38 下午 org.apache.catalina.startup.HostConfig deployDirectory
信息: Deployment of web application directory [F:\apache-tomcat-8.5.40\webapps\docs] has finished in [23] ms
五月 06, 2019 10:49:38 下午 org.apache.catalina.startup.HostConfig deployDirectory
信息: Deploying web application directory [F:\apache-tomcat-8.5.40\webapps\examples]
五月 06, 2019 10:49:39 下午 org.apache.catalina.core.ApplicationContext log
信息: ContextListener: contextInitialized()
五月 06, 2019 10:49:39 下午 org.apache.catalina.core.ApplicationContext log
信息: SessionListener: contextInitialized()
五月 06, 2019 10:49:39 下午 org.apache.catalina.core.ApplicationContext log
信息: ContextListener: attributeAdded('StockTicker', 'async.Stockticker@53ae374')
五月 06, 2019 10:49:39 下午 org.apache.catalina.startup.HostConfig deployDirectory
信息: Deployment of web application directory [F:\apache-tomcat-8.5.40\webapps\examples] has finished in [408] ms
五月 06, 2019 10:49:39 下午 org.apache.catalina.startup.HostConfig deployDirectory
信息: Deploying web application directory [F:\apache-tomcat-8.5.40\webapps\host-manager]
五月 06, 2019 10:49:39 下午 org.apache.catalina.startup.HostConfig deployDirectory
信息: Deployment of web application directory [F:\apache-tomcat-8.5.40\webapps\host-manager] has finished in [35] ms
五月 06, 2019 10:49:39 下午 org.apache.catalina.startup.HostConfig deployDirectory
信息: Deploying web application directory [F:\apache-tomcat-8.5.40\webapps\manager]
五月 06, 2019 10:49:39 下午 org.apache.catalina.startup.HostConfig deployDirectory
信息: Deployment of web application directory [F:\apache-tomcat-8.5.40\webapps\manager] has finished in [28] ms
五月 06, 2019 10:49:39 下午 org.apache.catalina.startup.HostConfig deployDirectory
信息: Deploying web application directory [F:\apache-tomcat-8.5.40\webapps\ROOT]
五月 06, 2019 10:49:39 下午 org.apache.catalina.startup.HostConfig deployDirectory
信息: Deployment of web application directory [F:\apache-tomcat-8.5.40\webapps\ROOT] has finished in [21] ms
五月 06, 2019 10:49:39 下午 org.apache.coyote.AbstractProtocol start
信息: Starting ProtocolHandler ["http-nio-8080"]
五月 06, 2019 10:49:39 下午 org.apache.coyote.AbstractProtocol start
信息: Starting ProtocolHandler ["ajp-nio-8009"]
五月 06, 2019 10:49:39 下午 org.apache.catalina.startup.Catalina start
信息: Server startup in 910 ms

效果出来了,tomcat服务器还没有看到启动成功,这个servlet就调用了构造方法和init初始化方法。

 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值