Servlet生命周期
实例化:Servlet容器创建Servlet的实例
Web容器负责加载Servlet,当web容器启动时或者在第一次使用这个Servlet的时候,容器会负责创建Servlet实例,但是用户必须通过web.xml指定Servlet的位置,成功加载后,Web容器会通过反射的方式对Servlet进行实例化
初始化:该容器调用init(ServletConfig)方法
当一个servlet被实例化后,容器将调用init()方法初始化这个对象,以便请求之前完成一些动作
服务:如果请求Servlet,则容器调用service()方法
当有请求提交时,将会先调用Servlet的service方法里实现逻辑,就不用调用父类方法。否则,还会先转发到相应的get/post方法里执行后在调用service方法里自己的逻辑
销毁:销毁实例之前调用destroy()方法
大部分Web容器关闭或者检测到一个Servlet要从容器中被删除时,会自动调用destroy方法
Servlet实例化
package com.ape.view;
import javax.servlet.ServletContext;
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
import javax.servlet.annotation.WebListener;
/**
* Application Lifecycle Listener implementation class DemoListener
*
*/
@WebListener
public class DemoListener implements ServletContextListener {
/**
* @see ServletContextListener#contextDestroyed(ServletContextEvent)
*/
public void contextDestroyed(ServletContextEvent sce) {
System.out.println("我被销毁了");
}
/**
* @see ServletContextListener#contextInitialized(ServletContextEvent)
*/
public void contextInitialized(ServletContextEvent sce) {
System.out.println("我被创建了");
}
}
Servlet的线程安全
在Servlet的生命周期中,构造方法只被执行一次。也就是说,在Servlet的整个生命周期中,
只存在一个Servlet实例对象。这说明Servlet是单例多线程的,可能会引起线程安全问题。
所谓线程安全就是一个Servlet实例对象会同时处理多个请求,这样的Servlet工作效率的确很
高,但如果Servlet中包含成员变量的话,多个线程同时操作变量,可能会引起线程安全问题。
Servlet线程安全的解决方法
1.添加锁
添加synchronized锁,并发访问就不存在线程安全问题,但是synchronized锁是一个线程访问时,其它线程只能等待上一个线程操作完后获取锁对象,进行操作,这也会大大影响效率。
2.使用局部变量
多线程并不会共享局部变量,使用Servlet中合理使用局部变量可以避免线程安全问题。