servlet介绍(一):https://www.cnblogs.com/xdp-gacl/p/3760336.html
servlet简单介绍
实现servlet接口的java类我们约定成俗称其为servlet
我们在实际使用中,一般继承的是HttpServlet,HttpServlet的service()方法会根据请求方式的不同,而调用doGet()或doPost()方法。所以继承了HttpServlet类之后,一般不需要重写service()方法,只需要重写doGet()或doPost()方法。
servlet的生命周期
在web.xml如果在<servlet>元素中配置了一个<load-on-startup>1</load-on-startup>元素,则servlet随项目的启动而创建,并执行init()方法初始化,随项目的关闭而销毁;如果没有,则当接受到了客户端的servlet请求之后,才会创建。
web服务端收到了一个客户端的servlet访问请求之后:
1.首先web服务端会判断该servelt是否已经实例化了一个对象,如果实例化的话,直接跳到第4步
2.实例化一个servlet对象
3.调用对象的init()方法初始化
4.调用对象的service()方法执行业务
5.服务器关闭时,会调用对象的distroy()方法销毁
servlet究竟是单例还是多例?
servlet容器默认采用单例多线程方式
但实际,在web.xml 中,声明servlet对应一个实例,生命几次就产生几个实例,即使是同个servelt,声明了多次,也会产生多个实例。
而如果一个servlet实现了singleThreadModel接口(标记接口),则会产生多个实例,但最多20个
servlet的线程安全问题
当多个客户端并发请求同个servlet时,web服务端会为每个请求创建一个线程,然后调用同时调用servlet实例对象的service()方法,因此当service()方法中有公共资源的话,会存在线程安全问题。
其中一个解决方法是用synchronized关键字对公共资源加锁,这个方法虽然可以解决线程安全问题,但是假如同时有1000人访问该servlet,只有一个人能占用公共资源,其他999人必须排队访问,这万万是不可取的。
另一个做法是,servlet实现singleThreadModel接口,该接口是一个标记接口,会使servlet引擎创建多个实例对象,并发的每个线程单独调用一个实例对象。这个方法严格来讲并没有真正解决多线程安全问题,因为真正意义的解决多线程问题是单个实例被多个线程同时调用的问题,所以,在Servlet API 2.4中,已经将SingleThreadModel标记为Deprecated(过时的)。