1. 程式人生 > >Servlet生命週期和工作原理的精彩講解

Servlet生命週期和工作原理的精彩講解

servlet生命週期和工作原理精彩講解!!

Servlet生命週期分為三個階段:

  1,初始化階段  呼叫init()方法

  2,響應客戶請求階段  呼叫service()方法

  3,終止階段  呼叫destroy()方法

Servlet初始化階段:

  在下列時刻Servlet容器裝載Servlet:

    1,Servlet容器啟動時自動裝載某些Servlet,實現它只需要在web.XML檔案中的<Servlet></Servlet>之間新增如下程式碼:

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

    2,在Servlet容器啟動後,客戶首次向Servlet傳送請求

    3,Servlet類檔案被更新後,重新裝載Servlet

  Servlet被裝載後,Servlet容器建立一個Servlet例項並且呼叫Servlet的init()方法進行初始化。在Servlet的整個生命週期內,init()方法只被呼叫一次。

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語言,擅長於處理流程和業務邏輯,缺點是生成動態網頁不直觀。

下面配合servlet例項會理解的更加清楚

1.在web.xml檔案

 <servlet>
  <display-name>Register</display-name>
  <servlet-name>Register</servlet-name>
  <servlet-class>org.cims201.ljk.servlet.RegisterServlet</servlet-class>  <!--Servlet類的位置-->
 </servlet>
 <servlet-mapping>
  <servlet-name>Register</servlet-name>
  <url-pattern>/Register</url-pattern>
 </servlet-mapping>

2.RegisterServlet類

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

public class RegisterServlet extends HttpServlet {


 protected void doGet(HttpServletRequest request,HttpServletResponse response)throws ServletException{
  System.out.println("呼叫doGet方法");
  response.setContentType("text;html;charset=utf-8");
  String username=request.getParameter("username");
  String password=request.getParameter("password");
  System.out.println("username="+username);
  System.out.println("password="+password);
  System.out.println("servlet test success....");
 }


protected void doPost(HttpServletRequest request,HttpServletResponse response)throws ServletException{
 System.out.println("呼叫doPost方法");
 doGet(request,response);
 }
/*
 http協議的格式主要有3中,get、post、head
 head一般不用再servlet中,get和post都可以用於資料傳遞,都可以把資料傳給伺服器,
 不同的是,他們的資料在http包中的位置不同,
 一般,get方法的資料放在頭部,直接在url裡面給出,
 而post在專門的資料區,因此post能傳遞大量的資料,一般在表單中可以選擇post或者get方法,
 
 用url直接傳參預設使用get

因為有這樣的不同,所以servlet中專門有doGet和doPost對應於這兩種傳參方式,
比如url = 
www.xxxx.com?aaa=b這樣的就是get方式傳參,
你要想得到aaa的值,必須重寫doGet方法,從request中得到值,從post中是得不到的。。。

實際上還有一個service方法,綜合了上面兩個,直接用這個是最好的了。。

 */
}

3.在URL位址列輸入:localhost:8080/myproject/Register?username=林&password=111

   輸出呼叫

doGet方法
username=??
password=12111217
servlet test success....

出現中文亂碼問題

4.處理中文亂碼問題

在Sertlet中加

response.setContentType("text/html;charset=utf-8")

在jsp頁面中加

<%@ page language="java" import="java.util.*,java.net.URLEncoder" pageEncoding="UTF-8"%>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">

在servlet中

String username=request.getParameter("username");

username=new String(username.getBytes(“ISO-8859-1”),"UTF-8");

最好的方法是通過Filter,但是在URL中傳中文,filter不管用。所以一般不再URL中傳中文,如果要傳的話,需要修改tomcatserver.xml檔案

<filter>
  <filter-name>Set Character Encoding</filter-name>
  <filter-class>
   org.cims201.util.SetCharacterEncodingFilter
  </filter-class>
  <init-param>
   <param-name>encoding</param-name>
   <param-value>UTF-8</param-value>
  </init-param>
  <init-param>
   <param-name>ignore</param-name>
   <param-value>true</param-value>
  </init-param>
 </filter>

<filter-mapping>
  <filter-name>Set Character Encoding</filter-name>
  <url-pattern>/*</url-pattern>
 </filter-mapping>
   <filter-mapping>

5標籤

<%@ include file="/common/taglibs.jsp"%>

taglibs.jsp內容

<%@ page pageEncoding="UTF-8"%>
<%@ taglib uri="/tags/struts-bean" prefix="bean" %>
<%@ taglib uri="/tags/struts-html" prefix="html" %>
<%@ taglib uri="/tags/struts-logic" prefix="logic" %>
<%@ taglib uri="/tags/struts-tiles" prefix="tiles" %>
<%@ taglib uri="/tags/extremecomponents" prefix="ec" %>
<%@ taglib uri="/tags/c" prefix="c" %>
<%@ taglib uri="/tags/tree" prefix="tree" %>
<%@ taglib uri="/tags/request" prefix="request" %>
<%@ taglib uri="/tags/fn" prefix="fn" %>
<%@ taglib uri="/tags/fmt" prefix="fmt" %>

6.錯誤跳轉頁面

<error-page>
 <error-code>500</error-code><!-- web伺服器內部錯誤 -->
 <location>/error.jsp</location>
</error-page>
 <error-page>
 <error-code>404</error-code><!--未找到資源 -->
 <location>/notfound.jsp</location>
</error-page>