文章目录
- 一、生命周期
- 二、错误实例
- 三、修改Servlet的初始化时机
一、生命周期
生命周期:实例化、初始化、服务、销毁
从出生到死亡的过程。对应Servlet中的三个方法:init(),service(),destroy()
-
被创建:执行init方法,只执行一次
默认情况下,第一次接收请求时,这个Servlet会进行实例化(在底层使用反射,调用构造方法)、初始化(init()方法)、然后服务(service)
-
提供服务:执行service方法,执行多次
从第二次请求开始,每一次都是服务
-
被销毁:执行destroy方法,只执行一次
随着tomcat容器的销毁,其中的所有servlet实例会被销毁,并调用销毁方法
结论:Servlet实例tomcat只会创建一个,所有的请求都是这个实例去响应。
二、错误实例
Demo02Servlet.java
// 注意这里改成了private
private Demo02Servlet () {
System.out.println("正在实例化...");
}
默认情况下,第一次请求时,tomcat才会去实例化,初始化,然后再服务。这样的好处是提高系统的启动速度。缺点是:第一次请求时,耗时较长。
结论:如果需要提高系统的启动速度,当前默认情况就是这样。如果需要提高响应速度,我们应该设置Servlet的初始化时机。
三、修改Servlet的初始化时机
默认是第一次请求时实例化,初始化。我们可以通过(startup:启动,load:加载)
可以配置指定Servlet启动的先后顺序,在标签下配置
的值为0或正整数(一般是0-10之间,最小值是0),数字越小,启动的时间越靠前
<servlet>
<servlet-name>Demo02Servlet</servlet-name>
<servlet-class>com.atguigu.servlets.Demo02Servlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
Servlet在容器中是单例(Demo02Servlet在tomcat容器里只会创建一个实例,即所有的请求都是同一个实例去响应)的、线程不安全的(一个线程需要根据这个实例中的某个成员变量值去做逻辑判断,但是在中间某个时机,从而导致第一个线程的执行路径发生了变化)。
线程1在读取num=1的时候,进入了service,刚刚准备执行if这个判断,但这个瞬间,线程2执行不同路径的代码,将num的值修改为了5,这个if就不成立了,就把线程1执行的路径修改掉了。
因为Servlet是单例的,所以所有的线程都是单例的,就会出现线程不安全。
我们已经知道了servlet是线程不安全的,给我们的启发是:尽量的不要在servlet中定义成员变量。如果不得不定义成员变量,那么:
①不要去修改成员变量的值
②不要根据成员变量的值做一些逻辑判断。