描述Servlet生命周期,Servlet是线程安全的吗? 为什么?

今天讲的是Servlet生命周期和Servlet的线程安全性问题

目录

一,Servlet的生命周期 

1. init() 方法

2. service() 方法

3. doGet() & doPost() 方法

4.destroy() 方法

二,Servlet的线程安全性


一,Servlet的生命周期 

Servlet 生命周期可被定义为从创建直到毁灭的整个过程。

        四个步骤即:实例化------>初始化------>服务------>销毁

  • 实例化        Servlet 容器创建 Servlet 的实例
  • 初始化        该容器调用 init (ServletConfig) 方法
  • 服务            如果请求 Servlet,则容器调用 service() 方法
  • 销毁            调用 destroy() 方法,最后由 JVM 的垃圾回收器进行垃圾回收

Servlet的生命周期如下:

1. init() 方法

        调用Servlet对象的init()方法,初始化Servlet的信息,init()方法只会在创建后被调用一次(初始化阶段)

  • init 方法被设计成只调用一次。它在第一次创建 Servlet 时被调用,在后续每次用户请求时不再调用。因此,它是用于一次性初始化,创建于用户第一次调用该 Servlet 的 URL 时,也可以指定 Servlet 在服务器第一次启动时被加载。
  • 当用户调用一个 Servlet 时,就会创建一个 Servlet 实例,每一个用户请求都会产生一个新的线程,适当的时候移交给 doGet 或 doPost 方法。
  • init() 方法简单地创建或加载一些数据,这些数据将被用于 Servlet 的整个生命周期。

2. service() 方法

        响应请求,调用service()或者是doGet(),doPost()方法来处理请求,这些方法是运行的在多线程状态下的。(响应客户请求阶段)

  • service() 方法是执行实际任务的主要方法,处理来自客户端(浏览器)的请求,并把格式化的响应写回给客户端。
  • 每次服务器接收到一个 Servlet 请求时,服务器会产生一个新的线程并调用服务。
  • Servlet接收和响应客户请求的过程,首先客户发送一个请求,Servlet是调用service()方法对请求进行响应的,通过源代码可见,service()方法中对请求的方式进行了匹配,选择调用doGet,doPost等这些方法,然后再进入对应的方法中调用逻辑层的方法,实现对客户的响应。由于在Servlet接口和GenericServlet中是没doGet,doPost等等这些方法的,HttpServlet中定义了这些方法,所以,我们每次定义一个Servlet的时候,都必须实现doGet或 doPost等这些方法。
  • service() 方法检查 HTTP 请求类型(GET、POST、PUT、DELETE 等),并在适当的时候调用 doGet、doPost、doPut,doDelete 等方法进行处理。

        要注意的是:由于service() 方法由容器调用,service 方法在适当的时候会调用 doGet、doPost、doPut、doDelete 等方法。所以,我们不用对 service() 方法做任何处理,只需要根据客户端的请求类型来重写 doGet() 或 doPost() 即可。 

3. doGet() & doPost() 方法

        用于处理请求,方法是运行的在多线程状态下的。(响应客户请求阶段)

  • 当URL的请求类型为GET请求 ,或者是一个未指定请求类型HTML表单,那么该请求将由 doGet() 方法处理。
  • 当请求来自一个特别指定了 请求类型为POST的 HTML 表单,那么该请求将由doPost() 方法处理。

4.destroy() 方法

        在长时间没有被调用或者是服务器关闭时,会调用destroy()方法来销毁Servlet对象。(终止阶段)

  • destroy() 方法只会被调用一次,在 Servlet 生命周期结束时被调用。
  • destroy() 方法可以让Servlet 关闭数据库连接、停止后台线程、把 Cookie 列表或点击计数器写入到磁盘,并执行其他类似的清理活动。
  • 在调用 destroy() 方法之后,servlet 对象被标记为垃圾回收。

二,Servlet的线程安全性

        首先,Servlet不是线程安全的

        要解释为什么Servlet为什么不是线程安全的,需要了解Servlet容器(即Tomcat)使如何响应HTTP请求的。

        当Tomcat接收到Client的HTTP请求时,Tomcat从线程池中取出一个线程,之后找到该请求对应的Servlet对象。如果该Servlet还未被请求过,那么将进行Servlet初始化并调用Servlet并调用service()方法。否则,直接调用service()方法。要注意的是每一个Servlet对象再Tomcat容器中只有一个实例对象,即是单例模式。如果多个HTTP请求请求的是同一个Servlet,那么着两个HTTP请求对应的线程将并发调用Servlet的service()方法。

        这时候,如果在Servlet中定义了实例变量或静态变量,那么可能会发生线程安全问题(因为所有的线程都可能使用这些变量)。

  • 6
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值