Servlet 的生命週期 和 工作原理
//轉載自 http://www.cnblogs.com/cuiliang/archive/2011/10/21/2220671.html 作者:艾澤拉斯的太陽
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語言,擅長於處理流程和業務邏輯,缺點是生成動態網頁不直觀。