Servlet容器工作原理深入講解
Servlet容器工作原理中,有兩個Servlet容器,***個很簡單,第二個則是根據(jù)***個寫出。為了使***個容器盡量簡單,所以沒有做得很完整。復(fù)雜一些的Servlet容器(包括TOMCAT4和5)在TOMCAT運行內(nèi)幕的其他章節(jié)有介紹。
兩個Servlet容器都處理簡單的Servlet及staticResource。您可以使用 webroot/ 目錄下的 PrimitiveServlet 來測試它。復(fù)雜一些的Servlet會超出這些容器的容量,您可以從 TOMCAT 運行內(nèi)幕 一書學(xué)習(xí)創(chuàng)建復(fù)雜的Servlet 容器。
兩個應(yīng)用程序的類都封裝在ex02.pyrmont 包下。在理解應(yīng)用程序如何運作之前,您必須熟悉 javax.servlet.Servlet 接口。首先就來介紹這個接口。隨后,就介紹servlet容器服務(wù)servlet的具體內(nèi)容。
javax.servlet.Servlet 接口
Servlet編程,需要引用以下兩個類和接口:javax.servlet 和 javax.servlet.http,在這些類和接口中, javax.servlet.Servlet接口尤為重要。所有的Servlet 必須實現(xiàn)這個接口或繼承已實現(xiàn)這個接口的類。
Servlet 接口有五個方法,如下
- public void init(ServletConfig config) throws ServletException
- public void service(ServletRequest request, ServletResponse response)
throws ServletException, java.io.IOException- public void destroy()
- public ServletConfig getServletConfig()
- public java.lang.String getServletInfo()
init、 service和 destroy 方法是 Servlet 生命周期的方法。當(dāng) Servlet 類實例化后,容器加載 init,以通知 servlet 它已進(jìn)入服務(wù)行列。init 方法必須被加載,Servelt 才能接收和請求。如果要載入數(shù)據(jù)庫驅(qū)動程序、初始化一些值等等,程序員可以重寫這個方法。在其他情況下,這個方法一般為空。
service 方法由 Servlet 容器調(diào)用,以允許 Servlet 響應(yīng)一個請求。Servlet 容器傳遞 javax.servlet.ServletRequest 對象和 javax.servlet.ServletResponse 對象。ServletRequest 對象包含客戶端 HTTP 請求信息, ServletResponse 則封裝servlet 響應(yīng)。這兩個對象,您可以寫一些需要 servlet 怎樣服務(wù)和客戶怎樣請求的代碼。
從service 中刪除Servlet實例之前,Servlet容器調(diào)用destroy方法。在servlet容器關(guān)閉或servlet容器需要更多的內(nèi)存時,就調(diào)用它。這個方法只有在 servlet的 service方法內(nèi)的所有線程都退出的時候,或在超時的時候才會被調(diào)用。在 servlet 容器調(diào)用 destroy方法之后,它將不再調(diào)用servlet的service方法。destroy 方法給了 servlet 機(jī)會,來清除所有候住的資源(比如:內(nèi)存,文件處理和線程),以確保在內(nèi)存中所有的持續(xù)狀態(tài)和 servlet的當(dāng)前狀態(tài)是同步的。Listing 2.1 包含了PrimitiveServlet 的代碼,此Servlet非常簡單,您 可以用它來測試本文中的Servlet容器應(yīng)用程序。
PrimitiveServlet 類實現(xiàn)了 javax.servlet.Servlet 并提供了五個servlet方法的接口 。它做的事情也很簡單:每次調(diào)用 init,service 或 destroy方法的時候,servlet就向控制口寫入方法名。service 方法也從ServletResponsec對象中獲得 java.io.PrintWriter 對象,并發(fā)送字符串到瀏覽器。
- import javax.servlet.*;
- import java.io.IOException;
- import java.io.PrintWriter;
- public class PrimitiveServlet implements Servlet {
- public void init(ServletConfig config) throws ServletException {
- System.out.println("init");
- }
- public void service(ServletRequest request, ServletResponse response) throws ServletException, IOException {
- System.out.println("from service");
- PrintWriter out = response.getWriter();
- out.println("Hello.Roses are red.");
- out.print("Violets are blue.");
- }
- public void destroy() {
- System.out.println("destroy");
- }
- public String getServletInfo() {
- return null;
- }
- public ServletConfig getServletConfig() {
- return null;
- }
- }
現(xiàn)在,我們從Servlet容器的角度來看看 servlet 編程。一個功能健全的Servlet容器對于每個Servlet 的HTTP請求會完成以下事情:
當(dāng)Servlet ***次被調(diào)用的時候,加載了Servlet類并調(diào)用它的init方法(僅調(diào)用一次)
響應(yīng)每次請求的時候 ,構(gòu)建一個javax.servlet.ServletRequest 和 javax.servlet.ServletResponse實例。
激活servlet的service方法,傳遞 ServletRequest 和 ServletResponse 對象。
當(dāng)servlet類關(guān)閉的時候,調(diào)用Servlet的destroy方法,并卸載servlet類。
發(fā)生在Servlet 容器內(nèi)部的事就復(fù)雜多了。只是這個簡單的servlet容器的功能不很健全,所以,這它只能運行非常簡單的servelt ,并不能調(diào)用servlet的init和destroy方法。然而,以下動作也被理解成Servlet容器的工作原理被執(zhí)行:
◆等待HTTP請求。
◆構(gòu)建ServletRequest和ServletResponse對象
如果請求的是一個staticResource,就會激活StaticResourceProcessor實例的 process方法,傳遞ServletRequest 和 ServletResponse 對象。
如果請求的是一個Servlet ,載入該類,并激活它的service方法,傳遞ServletRequest和ServletResponse 對象。注意:在這個servlet 容器,每當(dāng) servlet被請求的時候該類就被載入。
在***個應(yīng)用程序中,Servlet容器由六個類組成 。
◆HttpServer1
◆Request
◆Response
◆StaticResourceProcessor
◆ServletProcessor1
◆Constants
證如前文中的應(yīng)用程序一樣,這個程序的進(jìn)入口(靜態(tài) main 方法)是HttpServer 類。這個方法創(chuàng)建了HttpServer實例,并調(diào)用它的 await方法。這個方法等待 HTTP 請示,然后創(chuàng)建一個 request 對象和 response對象,根據(jù)請求是否是 staticResource還是 servlet 來分派它們到 StaticResourceProcessor實例或 ServletProcessor實例。
Constants 類包含 static find WEB_ROOT,它是從其他類引用的。 WEB_ROOT 指明 PrimitiveServlet 位置 和容器服務(wù)的staticResource。 HttpServer1 實例等待 HTTP 請求,直到它收到一個 shutdown 命令。發(fā)布 shutdown命令和前文是一樣的。 這就是Servlet容器工作原理,基于這個原理,眾多的Servlet應(yīng)用被執(zhí)行并返回我們期待的結(jié)果。
【編輯推薦】