老杜 JavaWeb 讲解(六) ——Servlet对象的生命周期

史前蓝熊 / 2023-05-14 / 原文

(八)Servlet对象的生命周期

对应视频:

10-Servlet对象的生命周期

8.1 什么是Servlet对象生命周期?

  • Servlet对象什么时候被创建。
  • Servlet对象什么时候被销毁。
  • Servlet对象创建了几个?
  • Servlet对象的生命周期表示:一个Servlet对象从出生在最后的死亡,整个过程是怎样的。

8.2 Servlet对象是由谁来维护的?

  • Servlet对象的创建,对象上方法的调用,对象最终的销毁,Javaweb程序员是无权干预的。
  • Servlet对象的生命周期是由Tomcat服务器(WEB Server)全权负责的。
  • Tomcat服务器通常我们又称为:WEB容器。(这个叫法你要知道【WEB Container】)
  • WEB容器来管理Servlet对象的死活。

8.3 思考:我们自己new的Servlet对象受WEB容器的管理吗?

  • 我们自己new的Servlet对象是不受WEB容器管理的。
  • WEB容器创建的Servlet对象,这些Servlet对象都会被放到一个集合当中(HashMap),只有放到这个HashMap集合中的Servlet才能够被WEB容器管理,自己new的Servlet对象不会被WEB容器管理。(自己new的Servlet对象不在容器当中)
  • web容器底层应该有一个HashMap这样的集合,在这个集合当中存储了Servlet对象和请求路径之间的关系

8.4 研究:服务器在启动的Servlet对象有没有被创建出来(默认情况下)?

  1. 在Servlet中提供一个无参数的构造方法,启动服务器的时候看看构造方法是否执行。
  2. 经过测试得出结论:默认情况下,服务器在启动的时候Servlet对象并不会被实例化。

这个设计是合理的。用户没有发送请求之前,如果提前创建出来所有的Servlet对象,必然是耗费内存的,并且创建出来的Servlet如果一直没有用户访问,显然这个Servlet对象是一个废物,没必要先创建。

怎么让服务器启动的时候创建Servlet对象呢?

  • 在servlet标签中添加子标签,在该子标签中填写整数,越小的整数优先级越高。

  • <servlet>
        <servlet-name>aservlet</servlet-name>
        <servlet-class>com.bjpowernode.javaweb.servlet.AServlet</servlet-class>
        <load-on-startup>1</load-on-startup>
    </servlet>
    <servlet-mapping>
        <servlet-name>aservlet</servlet-name>
        <url-pattern>/a</url-pattern>
    </servlet-mapping>
    

8.5 Servlet对象生命周期

第一步: 默认情况下服务器启动的时候AServlet对象并没有被实例化

第二步:用户发送第一次请求的时候,控制台输出了以下内容:

AServlet无参数构造方法执行了
AServlet's init method execute!
AServlet's service method execute!

根据以上输出内容得出结论:

  • 用户在发送第一次请求的时候Servlet对象被实例化(AServlet的构造方法被执行了。并且执行的是无参数构造方法。)

  • AServlet对象被创建出来之后,Tomcat服务器马上调用了AServlet对象的init方法。(init方法在执行的时候,AServlet对象已经存在了。已经被创建出来了。)

  • 用户发送第一次请求的时候,init方法执行之后,Tomcat服务器马上调用AServlet对象的service方法。

第三步:用户继续发送第二次请求,控制台输出了以下内容:

AServlet's service method execute!

根据以上输出结果得知,用户在发送第二次,或者第三次,或者第四次请求的时候,Servlet对象并没有新建,还是使用之前创建好的Servlet对象,直接调用该Servlet对象的service方法,这说明:

  • Servlet对象是单例的(单实例的。但是要注意:Servlet对象是单实例的,但是Servlet类并不符合单例模式。我们称之为假单例。之所以单例是因为Servlet对象的创建我们javaweb程序员管不着,这个对象的创建只能是Tomcat来说了算,Tomcat只创建了一个,所以导致了单例,但是属于假单例。真单例模式,构造方法是私有化的。)

  • 无参数构造方法、init方法只在第一次用户发送请求的时候执行。也就是说无参数构造方法只执行一次。init方法也只被Tomcat服务器调用一次。

  • 只要用户发送一次请求:service方法必然会被Tomcat服务器调用一次。发送100次请求,service方法会被调用100次。

第四步:关闭服务器的时候,控制台输出了以下内容:

AServlet's destroy method execute!

通过以上输出内容,可以得出以下结论:

  • Servlet的destroy方法只被Tomcat服务器调用一次。

  • destroy方法是在什么时候被调用的?

    • 在服务器关闭的时候。
    • 因为服务器关闭的时候要销毁AServlet对象的内存。
    • 服务器在销毁AServlet对象内存之前,Tomcat服务器会自动调用AServlet对象的destroy方法。

destroy方法调用的时候,对象销毁了还是没有销毁呢?

答:destroy方法执行的时候AServlet对象还在,没有被销毁。destroy方法执行结束之后,AServlet对象的内存才会被Tomcat释放。

第五步:Servlet对象更像一个人的一生:

  • Servlet的无参数构造方法执行:标志着你出生了。
  • Servlet对象的init方法的执行:标志着你正在接受教育。
  • Servlet对象的service方法的执行:标志着你已经开始工作了,已经开始为人类提供服务了。
  • Servlet对象的destroy方法的执行:标志着临终。有什么遗言,抓紧的。要不然,来不及了。

第六步:关于Servlet类中方法的调用次数?

  • 构造方法只执行一次。
  • init方法只执行一次。
  • service方法:用户发送一次请求则执行一次,发送N次请求则执行N次。
  • destroy方法只执行一次。