servlet的生命週期詳解

1、servlet生命週期原理解析html

一、Servlet生命週期分爲三個階段:java

  (1)初始化階段  調用init()方法web

  (2)響應客戶請求階段  調用service()方法tomcat

  (3)終止階段  調用destroy()方法安全

  對於Servlet初始化階段:在以下時刻Servlet容器裝載Servlet:服務器

  (1)Servlet容器啓動時自動裝載某些Servlet,實現它只須要在web.XML文件中的<Servlet></Servlet>之間添加以下代碼:app

<loadon-startup>1</loadon-startup> 

  (2)在Servlet容器啓動後,客戶首次向Servlet發送請求ide

  (3)Servlet類文件被更新後,從新裝載Servletpost

  Servlet被裝載後,Servlet容器建立一個Servlet實例而且調用Servlet的init()方法進行初始化。在Servlet的整個生命週期內,init()方法只被調用一次。url

二、Servlet工做原理:

  首先簡單解釋一下Servlet接收和響應客戶請求的過程,首先客戶發送一個請求,Servlet是調用service()方法對請求進行響應的,經過源代碼可見,service()方法中對請求的方式進行了匹配,選擇調用doGet,doPost等這些方法,而後再進入對應的方法中調用邏輯層的方法,實現對客戶的響應。在Servlet接口和GenericServlet中是沒有doGet,doPost等等這些方法的,HttpServlet中定義了這些方法,可是都是返回error信息,因此,咱們每次定義一個Servlet的時候,都必須實現doGet或doPost等這些方法。

  每個自定義的Servlet都必須實現Servlet的接口,Servlet接口中定義了五個方法,其中比較重要的三個方法涉及到Servlet的生命週期,分別是上文提到的init(),service(),destroy()方法。GenericServlet是一個通用的,不特定於任何協議的Servlet,它實現了Servlet接口。而HttpServlet繼承於GenericServlet,所以HttpServlet也實現了Servlet接口。因此咱們定義Servlet的時候只須要繼承HttpServlet便可。

  Servlet接口和GenericServlet是不特定於任何協議的,而HttpServlet是特定於HTTP協議的類,因此HttpServlet中實現了service()方法,並將請求ServletRequest,ServletResponse強轉爲HttpRequest和HttpResponse。

public void service(ServletRequest req,ServletResponse res) 
  throws ServletException,IOException
{
      HttpRequest request;
      HttpResponse response;
 
     try
     {
         req = (HttpRequest)request;
         res = (HttpResponse)response;
      }catch(ClassCastException e)
      {
         throw new ServletException("non-HTTP request response"); 
      }
      service(request,response);
}

  代碼的最後調用了HTTPServlet本身的service(request,response)方法,而後根據請求去調用對應的doXXX方法,由於HttpServlet中的doXXX方法都是返回錯誤信息。

protected void doGet(HttpServletRequest res,HttpServletResponse resp)
  throws ServletException,IOException
{
   String protocol = req.getProtocol();
   String msg = IStrings.getString("http.method_get_not_supported");
   if(protocol.equals("1.1"))
   {
      resp.sendError(HttpServletResponse.SC.METHOD.NOT.ALLOWED,msg);
    }
   esle
    {
      resp.sendError(HttpServletResponse.SC_BAD_REQUEST,msg);
    }
}

  因此須要咱們在自定義的Servlet中override這些方法!

三、Servlet響應請求階段:

  對於用戶到達Servlet的請求,Servlet容器會建立特定於這個請求的ServletRequest對象和ServletResponse對象,而後調用Servlet的service方法。service方法從ServletRequest對象得到客戶請求信息,處理該請求,並經過ServletResponse對象向客戶返回響應信息。

  對於Tomcat來講,它會將傳遞過來的參數放在一個Hashtable中,該Hashtable的定義是:

private Hashtable<String String[]> paramHashStringArray = new Hashtable<String String[]>();

  這是一個String-->String[]的鍵值映射。

  HashMap線程不安全的,Hashtable線程安全。

四、Servlet終止階段:

  當WEB應用被終止,或Servlet容器終止運行,或Servlet容器從新裝載Servlet新實例時,Servlet容器會先調用Servlet的destroy()方法,在destroy()方法中能夠釋放掉Servlet所佔用的資源。

五、Servlet什麼時候被建立: 

  (1)默認狀況下,當WEB客戶第一次請求訪問某個Servlet的時候,WEB容器將建立這個Servlet的實例。

  (2)當web.xml文件中若是<servlet>元素中指定了<load-on-startup>子元素時,Servlet容器在啓動web服務器時,將按照順序建立並初始化Servlet對象。

  注意:在web.xml文件中,某些Servlet只有<serlvet>元素,沒有<servlet-mapping>元素,這樣咱們沒法經過url的方式訪問這些Servlet,這種Servlet一般會在<servlet>元素中配置一個<load-on-startup>子元素,讓容器在啓動的時候自動加載這些Servlet並調用init()方法,完成一些全局性的初始化工做。

六、Web應用什麼時候被啓動:

  (1)當Servlet容器啓動的時候,全部的Web應用都會被啓動

  (2)控制器啓動web應用

七、Servlet與JSP的比較:

  有許多類似之處,均可以生成動態網頁。

  JSP的優勢是擅長於網頁製做,生成動態頁面比較直觀,缺點是不容易跟蹤與排錯。

  Servlet是純Java語言,擅長於處理流程和業務邏輯,缺點是生成動態網頁不直觀。

 

2、servlet生命週期實例

新建一個web工程

MyServlet.java

package com.demo.servlet;

import java.io.IOException;

import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

public class MyServlet extends HttpServlet {

    private static final long serialVersionUID = 8721524830738176364L;
    
    public MyServlet() {
        System.out.println("Constructor......");
    }
    
    @Override
    public void doPost(HttpServletRequest req, HttpServletResponse resp)
            throws ServletException, IOException {
        // TODO Auto-generated method stub
        System.out.println("doPost;");
    }

    @Override
    public void service(HttpServletRequest arg0, HttpServletResponse arg1)
            throws ServletException, IOException {
        // TODO Auto-generated method stub
        System.out.println("service;");
        super.service(arg0, arg1);
    }

    @Override
    public void destroy() {
        // TODO Auto-generated method stub
        System.out.println("destroy;");
        super.destroy();
    }

    @Override
    public void init() throws ServletException {
        // TODO Auto-generated method stub
        System.out.println("init;");
        super.init();
    }

    @Override
    public void init(ServletConfig config) throws ServletException {
        // TODO Auto-generated method stub
        System.out.println("init(ServletConfig);");
        super.init(config);
    }
}

web.xml

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
      xmlns
="http://java.sun.com/xml/ns/javaee"
      xmlns:web
="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
      xsi:schemaLocation
="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" version="3.0"> <display-name>servlet_demo</display-name> <servlet> <servlet-name>MyServlet</servlet-name> <servlet-class>com.demo.servlet.MyServlet</servlet-class> </servlet> <servlet-mapping> <servlet-name>MyServlet</servlet-name> <url-pattern>/MyServlet</url-pattern> </servlet-mapping> </web-app>

index.html

<html>  
  <body>  
    This is my Html page. <br>  
    <form action="MyServlet" method="post">  
    <input type=submit value="mySubmit">  
    </form>  
  </body>  
</html>  

而後啓動項目,日誌沒有打印相關servlet的信息。

咱們訪問index.html

http://127.0.0.1:8100/index.html

點擊mySubmit

  咱們能夠看到日誌輸出以下:

以後咱們繼續訪問http://127.0.0.1:8100/MyServlet,日誌以下:

而後咱們在tomcat下中止這個工程:

 

可是當咱們在web.xml中配置以下信息,重啓tomcat,

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
    xmlns="http://java.sun.com/xml/ns/javaee" 
    xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" 
    xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" version="3.0">
  <display-name>servlet_demo</display-name>
  
  <servlet>  
    <servlet-name>MyServlet</servlet-name>  
    <servlet-class>com.demo.servlet.MyServlet</servlet-class>  
    <load-on-startup>1</load-on-startup>
  </servlet>  
  <servlet-mapping>  
    <servlet-name>MyServlet</servlet-name>  
    <url-pattern>/MyServlet</url-pattern>  
  </servlet-mapping>  
</web-app>

tomcat啓動日誌以下:

由此能夠得出以下結論:servlet只被構造並初始化一次(第一次訪問的時候被構造和初始化),以後提交表單,servlet就只會相應一次dopost,當tomcat移除項目後,detroy會被調用,這個servlet終止;若是再web.xml中servlet配置了<load-on-startup>1</load-on-startup>,tomcat在啓動的時候就會加載這個servlet(實例化並調用其init()方法);在servlet整個生命週期中,初始化init() 和destroy()銷燬只會被調用一次。

相關文章
相關標籤/搜索