目录
一、概述
当我们的浏览器访问服务器时,服务器会对URL进行解析,然后根据URL访问对应的Servlet,以Tomcat为例,如果是第一次访问,Tomcat服务器会创建这个Servlet的实例对象,接着通过Java反射技术来自动调用Servlet的init()、service()等方法,最终,当Tomcat关闭或者重启时,Servlet就会被销毁,这是一个Servlet的完整生命周期。
二、图示展示Servlet生命周期
三、Servlet生命周期内的四步骤
1.加载和实例化
Servlet容器负责加载和实例化Servlet。当Servlet容器启动时,或者在容器检测到需要这个Servlet来响应第一个请求时,创建Servlet实例。当Servlet容器启动后,它必须要知道所需的Servlet类在什么位置,Servlet容器可以从本地文件系统、远程文件系统或者其他的网络服务中通过类加载器加载Servlet类,成功加载后,容器创建Servlet的实例。因为容器是通过Java的反射API来创建Servlet实例,调用的是Servlet的默认构造方法(即不带参数的构造方法),所以我们在编写Servlet类的时候,不应该提供带参数的构造方法。
1.Servlet实例第一次实例化代码示例:
@WebServlet("/home.do")
public class HomeServlet extends HttpServlet{
// 1.浏览器发送请求至服务器,服务器收到请求,创建HomeServlet的实例对象
public HomeServlet() {
System.out.println("Tomcat创建HomeServlet实例");
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws
ServletException, IOException {
}
}
}
我在这里连续请求了两次HomeServlet,但只有第一次才创建了Servlet实例。
2.初始化
在Servlet实例化之后,容器将调用Servlet的init()方法初始化这个对象。初始化的目的是为了让Servlet对象在处理客户端请求前完成一些初始化的工作,如建立数据库的连接,获取配置信息等。对于每一个Servlet实例,init()方法只被调用一次。在初始化期间,Servlet实例可以使用容器为它准备的ServletConfig对象从Web应用程序的配置信息(在web.xml中配置)中获取初始化的参数信息。在初始化期间,如果发生错误,Servlet实例可以抛出ServletException异常或者UnavailableException异常来通知容器。ServletException异常用于指明一般的初始化失败,例如没有找到初始化参数;而UnavailableException异常用于通知容器该Servlet实例不可用。例如,数据库服务器没有启动,数据库连接无法建立,Servlet就可以抛出UnavailableException异常向容器指出它暂时或永久不可用。
2.Servlet初始化代码示例:
@WebServlet("/home.do")
public class HomeServlet extends HttpServlet{
// 1.浏览器发送请求至服务器,服务器收到请求,创建HomeServlet的实例对象
public HomeServlet() {
System.out.println("Tomcat创建HomeServlet实例");
}
// 2.服务器初始化HomeServlet实例对象
@Override
public void init() throws ServletException {
System.out.println("服务器初始化HomeServlet对象");
}
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println("服务阶段");
}
}
初始化和创建实例对象一样,在一个Servlet生命周期内,它只会执行一次。
3.服务阶段
Servlet容器调用Servlet的service()方法对请求进行处理。要注意的是,在service()方法调用之前,init()方法必须成功执行。在service()方法中,Servlet实例通过ServletRequest对象得到客户端的相关信息和请求信息,在对请求进行处理后,调用ServletResponse对象的方法设置响应信息。在service()方法执行期间,如果发生错误,Servlet实例可以抛出ServletException异常或者UnavailableException异常。如果UnavailableException异常指示了该实例永久不可用,Servlet容器将调用实例的destroy()方法,释放该实例。此后对该实例的任何请求,都将收到容器发送的HTTP 404(请求的资源不可用)响应。如果UnavailableException异常指示了该实例暂时不可用,那么在暂时不可用的时间段内,对该实例的任何请求,都将收到容器发送的HTTP 503(服务器暂时忙,不能处理请求)响应。
3.Servlet服务阶段代码示例:
@WebServlet("/home.do")
public class HomeServlet extends HttpServlet{
// 1.浏览器发送请求至服务器,服务器收到请求,创建HomeServlet的实例对象
public HomeServlet() {
System.out.println("Tomcat创建HomeServlet实例");
}
// 2.服务器初始化HomeServlet实例对象
@Override
public void init() throws ServletException {
System.out.println("服务器初始化HomeServlet对象");
}
// 3.服务:服务器调用父类HTTPServlet的service()方法,根据请求方式的不同,调用子类的doGet()方法或doPost()方法
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println("父类的service方法执行");
}
}
Servlet生命周期内,每调用一次Servlet,service()方法就会执行一次。
4.销毁
当容器检测到一个Servlet实例应该从服务中被移除的时候,容器就会调用实例的destroy()方法,以便让该实例可以释放它所使用的资源,保存数据到持久存储设备中。当需要释放内存或者容器关闭时,容器就会调用Servlet实例的destroy()方法。在destroy()方法调用之后,容器会释放这个Servlet实例,该实例随后会被Java的垃圾收集器所回收。如果再次需要这个Servlet处理请求,Servlet容器会创建一个新的Servlet实例。
4.销毁阶段代码实现:
@WebServlet("/home.do")
public class HomeServlet extends HttpServlet{
// 1.浏览器发送请求至服务器,服务器收到请求,创建HomeServlet的实例对象
public HomeServlet() {
System.out.println("Tomcat创建HomeServlet实例");
}
// 2.服务器初始化HomeServlet实例对象
@Override
public void init() throws ServletException {
System.out.println("服务器初始化HomeServlet对象");
}
// 3.服务:服务器调用父类HTTPServlet的service()方法,根据请求方式的不同,调用子类的doGet()方法或doPost()方法
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println("父类的service方法执行");
}
// 4.销毁
@Override
public void destroy() {
System.out.println("执行销毁");
}
}
当处在一个Servlet生命周期内,多次调用service()并不会执行销毁Servlet实例的方法,只有当Tomcat服务器关闭时,才会执行销毁程序,再次重启Tomcat服务器,就是Servlet一个新的生命周期。
四、总结:
web容器加载Servlet并将其实例化后,其生命周期正式开始。容器首先运行init()方法进行Servlet的初始化:然后请求达到之后调用Servlet的service方法,service()方法会根据连接中的内容来调用具体的方法并且给出返回值。当服务器关闭的时候servlet实例将会被销毁,此时调用的是servlet的destory方法, init() and destory()方法都只会执行一次,service方法则是在每次请求servlet的时候都会执行。如果servlet中需要用到一些初始化与销毁的资源,可以将初始化的资源放到init()方法中,销毁资源的代码放在destory()方法中。