servlet 工作原理
本文介紹一個(gè)簡單 servlet 容器的基本原理。
Servlet容器工作原理講解本文介紹一個(gè)簡單 servlet 容器的基本原理。現(xiàn)有兩個(gè)servlet容器,第一個(gè)很簡單,第二個(gè)則是根據(jù)第一個(gè)寫出。為了使第一個(gè)容器盡量簡單,所以沒有做得很完整。復(fù)雜一些的 servlet容器(包括TOMCAT4和5)在TOMCAT運(yùn)行內(nèi)幕的其他章節(jié)有介紹。?
兩個(gè)servlet容器都處理簡單的 servlet及staticResource。您可以使用 webroot/ 目錄下的 PrimitiveServlet 來測試它。復(fù)雜一些的 servlet會(huì)超出這些容器的容量,您可以從 TOMCAT 運(yùn)行內(nèi)幕 一書學(xué)習(xí)創(chuàng)建復(fù)雜的 servlet 容器。?
兩個(gè)應(yīng)用程序的類都封裝在ex02.pyrmont 包下。在理解應(yīng)用程序如何運(yùn)作之前,您必須熟悉 javax.servlet.Servlet 接口。首先就來介紹這個(gè)接口。隨后,就介紹servlet容器服務(wù)servlet的具體內(nèi)容。?
javax.servlet.Servlet 接口 ?
servlet 編程,需要引用以下兩個(gè)類和接口:javax.servlet 和 javax.servlet.http,在這些類和接口中,javax.servlet.Servlet接口尤為重要。所有的 servlet 必須實(shí)現(xiàn)這個(gè)接口或繼承已實(shí)現(xiàn)這個(gè)接口的類。?
Servlet 接口有五個(gè)方法,如下?
| public void init(ServletConfig config) throws ServletException public void service(ServletRequest request, ServletResponse response) throws ServletException, java.io.IOExceptionpublic void destroy()public ServletConfig getServletConfig()public java.lang.String getServletInfo() |
init、 service和 destroy 方法是 Servlet 生命周期的方法。當(dāng) Servlet 類實(shí)例化后,容器加載 init,以通知 servlet 它已進(jìn)入服務(wù)行列。init 方法必須被加載,Servelt 才能接收和請求。如果要載入數(shù)據(jù)庫驅(qū)動(dòng)程序、初始化一些值等等,程序員可以重寫這個(gè)方法。在其他情況下,這個(gè)方法一般為空。?
service 方法由 Servlet 容器調(diào)用,以允許 Servlet 響應(yīng)一個(gè)請求。Servlet 容器傳遞 javax.servlet.ServletRequest 對象和 javax.servlet.ServletResponse 對象。ServletRequest 對象包含客戶端 HTTP 請求信息,ServletResponse 則封裝servlet 響應(yīng)。這兩個(gè)對象,您可以寫一些需要 servlet 怎樣服務(wù)和客戶怎樣請求的代碼。?
從service中刪除Servlet實(shí)例之 前,容器調(diào)用destroy方法。在servlet容器關(guān)閉或servlet容器需要更多的內(nèi)存時(shí),就調(diào)用它。這個(gè)方法只有在servlet的 service方法內(nèi)的所有線程都退出的時(shí)候,或在超時(shí)的時(shí)候才會(huì)被調(diào)用。在 servlet 容器調(diào)用 destroy方法之后,它將不再調(diào)用servlet的service方法。destroy 方法給了 servlet 機(jī)會(huì),來清除所有候住的資源(比如:內(nèi)存,文件處理和線程),以確保在內(nèi)存中所有的持續(xù)狀態(tài)和 servlet的當(dāng)前狀態(tài)是同步的。Listing 2.1 包含了PrimitiveServlet 的代碼,此servlet非常簡單,您 可以用它來測試本文中的servlet容器應(yīng)用程序。?
PrimitiveServlet 類實(shí)現(xiàn)了javax.servlet.Servlet 并提供了五個(gè)servlet方法的接口 。它做的事情也很簡單:每次調(diào)用 init,service 或 destroy方法的時(shí)候,servlet就向控制口寫入方法名。service 方法也從ServletResponsec對象中獲得java.io.PrintWriter 對象,并發(fā)送字符串到瀏覽器。?
| Listing 2.1.PrimitiveServlet.java 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;} } |
Application 1 ?
現(xiàn)在,我們從 servlet容器的角度來看看 servlet 編程。一個(gè)功能健全的 servlet容器對于每個(gè) servlet 的HTTP請求會(huì)完成以下事情:?
當(dāng)servlet 第一次被調(diào)用的時(shí)候,加載了 servlet類并調(diào)用它的init方法(僅調(diào)用一次)?
響應(yīng)每次請求的時(shí)候 ,構(gòu)建一個(gè)javax.servlet.ServletRequest 和 javax.servlet.ServletResponse實(shí)例。?
激活servlet的service方法,傳遞 ServletRequest 和 ServletResponse 對象。?
當(dāng)servlet類關(guān)閉的時(shí)候,調(diào)用servlet的destroy方法,并卸載servlet類。?
發(fā)生在 servlet 容器內(nèi)部的事就復(fù)雜多了。只是這個(gè)簡單的servlet容器的功能不很健全,所以,這它只能運(yùn)行非常簡單的servelt ,并不能調(diào)用servlet的init和destroy方法。然而,它也執(zhí)行了以下動(dòng)作:?
等待HTTP請求。?
構(gòu)建ServletRequest和ServletResponse對象?
如果請求的是一個(gè)staticResource,就會(huì)激活StaticResourceProcessor實(shí)例的 process方法,傳遞ServletRequest 和 ServletResponse 對象。?
如果請求的是一個(gè)servlet ,載入該類,并激活它的service方法,傳遞ServletRequest和ServletResponse 對象。注意:在這個(gè)servlet 容器,每當(dāng) servlet被請求的時(shí)候該類就被載入。?
在第一個(gè)應(yīng)用程序中,servlet容器由六個(gè)類組成 。?
HttpServer1?
Request?
Response?
StaticResourceProcessor?
ServletProcessor1?
Constants?
證 如前文中的應(yīng)用程序一樣,這個(gè)程序的進(jìn)入口(靜態(tài) main 方法)是HttpServer 類。這個(gè)方法創(chuàng)建了HttpServer實(shí)例,并調(diào)用它的await方法。這個(gè)方法等待 HTTP 請示,然后創(chuàng)建一個(gè) request 對象和 response對象,根據(jù)請求是否是staticResource還是 servlet 來分派它們到 StaticResourceProcessor實(shí)例或ServletProcessor實(shí)例。?
Constants 類包含 static find WEB_ROOT,它是從其他類引用的。 WEB_ROOT 指明 PrimitiveServlet 位置 和容器服務(wù)的staticResource。?
HttpServer1 實(shí)例等待 HTTP 請求,直到它收到一個(gè) shutdown 命令。發(fā)布 shutdown命令和前文是一樣的。?
Servlet容器工作原理講解(2)
HttpServer1 類?
此應(yīng)用程序內(nèi)的 HttpServer1類 與前文簡單的 WEB 服務(wù)器應(yīng)用程序中的HttpServer 十分相似。但是,此應(yīng)用程序內(nèi)的 HttpServer1 能服務(wù)靜態(tài)資源和 servlet。如果要請求一個(gè)靜態(tài)資源,請輸入以下 URL:?
http://machineName:port/staticResource?
它就是前文中提到的怎樣在 WEB 服務(wù)器應(yīng)用程序里請求靜態(tài)資源。如果要請求一個(gè) servlet,請輸入以下 URL:?
http://machineName:port/servlet/servletClass?
如果您想在本地瀏覽器請求一個(gè) PrimitiveServle servlet ,請輸入以下 URL:?
http://localhost:8080/servlet/PrimitiveServlet?
下面 Listing 2.2 類的 await 方法,是等待一個(gè) HTTP 請求,直到一個(gè)發(fā)布 shutdown 命令。與前文的 await 方法相似。?
| Listing 2.2. HttpServer1 類的 await 方法 public void await() { ServerSocket serverSocket = null; int port = 8080;try { serverSocket = new ServerSocket(port, 1, InetAddress.getByName("127.0.0.1"));} catch (IOException e) { e.printStackTrace(); System.exit(1);}// 循環(huán),等待一個(gè)請求 while (!shutdown) { Socket socket = null; InputStream input = null; OutputStream output = null;try { socket = serverSocket.accept(); input = socket.getInputStream(); output = socket.getOutputStream();// 創(chuàng)建請求對象并解析 Request request = new Request(input); request.parse();// 創(chuàng)建回應(yīng)對象 Response response = new Response(output); response.setRequest(request);//檢測是否是 servlet 或靜態(tài)資源的請求 //servlet 請求以 "/servlet/" 開始 if (request.getUri().startsWith("/servlet/")) { ServletProcessor1 processor = new ServletProcessor1(); processor.process(request, response);} else { StaticResourceProcessor processor = new StaticResourceProcessor(); processor.process(request, response);}// 關(guān)閉socket socket.close();//檢測是否前面的 URI 是一個(gè) shutdown 命令 shutdown = request.getUri().equals(SHUTDOWN_COMMAND);} catch (Exception e) { e.printStackTrace(); System.exit(1);}} } |
此文 await 方法和前文的不同點(diǎn)就是,此文的 await 方法中的請求調(diào)度到StaticResourceProcessor 或 ervletProcessor 。?
如果 URI中包含 "/servlet/.",請求推進(jìn)到后面,否則,請求傳遞到 StaticResourceProcessor 實(shí)例?
Request 類?
Servlet service 方法接受 servlet 容器的 javax.servlet.ServletRequest 和javax.servlet.ServletResponse 實(shí)例。因此,容器必須構(gòu)建 ServletRequest和ServletResponse對象,然后將其傳遞到正在被服務(wù)的service 方法。?
ex02.pyrmont.Request 類代表一個(gè)請求對象傳遞到 service 方法。同樣地,它必須實(shí)現(xiàn) javax.servlet.ServletRequest 接口。這個(gè)類必須提供接口內(nèi)所有方法的實(shí)現(xiàn)。這里盡量簡化它并只實(shí)現(xiàn)幾個(gè)方法。要編譯 Request 類的話,必須提供這些方法的空實(shí)現(xiàn)。再來看看 request 類,內(nèi)部所有需要返回一個(gè)對象實(shí)例都返回null,如下:?
| public Object getAttribute(String attribute) { return null;}public Enumeration getAttributeNames() { return null;}public String getRealPath(String path) { return null;} |
另外,request 類仍需有前文有介紹的 parse 和getUri 方法。?
Response 類?
response 類實(shí)現(xiàn) javax.servlet.ServletResponse,同樣,該類也必須提供接口內(nèi)所有方法的實(shí)現(xiàn)。類似于 Request 類,除 getWriter 方法外,其他方法的實(shí)現(xiàn)都為空。?
| public PrintWriter getWriter() { // autoflush is true, println() will flush, // but print() will not. writer = new PrintWriter(output, true); return writer;} |
PrintWriter 類構(gòu)建器的第二個(gè)參數(shù)是一個(gè)代表是否啟用 autoflush 布爾值 ,如果為真,所有調(diào)用println 方法都 flush 輸出。而 print 調(diào)用則不 flush 輸出。因此,如果在servelt 的service 方法的最后一行調(diào)用 print方法,則從瀏覽器上看不到此輸出 。這個(gè)不完整性在后面的應(yīng)用程序內(nèi)會(huì)有調(diào)整。?
response 類也包含有前文中介紹的 sendStaticResource方法。?
StaticResourceProcessor 類?
StaticResourceProcessor 類用于服務(wù)靜態(tài)資源的請求。它唯一的方法是 process。?
| Listing 2.3.StaticResourceProcessor 類的 process方法。 public void process(Request request, Response response) { try { response.sendStaticResource();} catch (IOException e) { e.printStackTrace();} } |
process 方法接受兩個(gè)參數(shù):Request 和 Response 實(shí)例。它僅僅是調(diào)用 response 類的 sendStaticResource 方法。
Servlet容器工作原理講解(3)
ServletProcessor1 類?
ServletProcessor1 類用來處理對 servlet 的 HTTP 請求。 它非常簡單,只包含了一個(gè) process 方法。 而這個(gè)方法接受兩個(gè)參數(shù): 一個(gè)javax.servlet.ServletRequest 實(shí)例和一個(gè) avax.servlet.ServletResponse實(shí)例。 process 方法也構(gòu)建了一個(gè) java.net.URLClassLoader 對象并使用它裝載 servlet 類文件。 在從類裝載器獲得的 Class 對象上,process 方法創(chuàng)建一個(gè) servlet 實(shí)例并調(diào)用它的 service 方法。?
process 方法?
Listing 2.4. ServletProcessor1 類中 process 方法?
| public void process(Request request, Response response) {String uri = request.getUri();String servletName = uri.substring(uri.lastIndexOf("/") + 1);URLClassLoader loader = null;try {// create a URLClassLoaderURLStreamHandler streamHandler = null;URL[] urls = new URL[1];File classPath = new File(Constants.WEB_ROOT);String repository = (new URL("file", null, classPath.getCanonicalPath() + File.separator)).toString() urls[0] = new URL(null, repository, streamHandler);loader = new URLClassLoader(urls);}catch (IOException e) {System.out.println(e.toString());}Class myClass = null;try {myClass = loader.loadClass(servletName);}catch (Exception e) {System.out.println(e.toString());}Servlet servlet = null;try {servlet = (Servlet) myClass.newInstance();servlet.service((ServletRequest) request, (ServletResponse) response);}catch (Exception e) {System.out.println(e.toString());}catch (Throwable e) {System.out.println(e.toString());} } |
process方法接受兩個(gè)參數(shù):一個(gè) ServletRequest實(shí)例和一個(gè) ServletResponse 實(shí)例。process方法通過調(diào)用 getRequestUri 方法從 ServletRequest獲取 URI。?
String uri = request.getUri();切記 URI 的格式:?
/servlet/servletName?
servletName是servlet類的名稱。?
如果要裝載 servlet 類,則需要使用以下代碼從 URI 獲知 servlet 名稱:String servletName = uri.substring(uri.lastIndexOf("/") + 1);然后 process 方法裝載 servlet。 要做到這些,需要?jiǎng)?chuàng)建一個(gè)類裝載器,并告訴裝載器該類的位置, 該 servlet 容器可以指引類裝載器在 Constants.WEB_ROOT 指向的目錄中查找。 在工作目錄下,WEB_ROOT 指向 webroot/ 目錄。?
如果要裝載一個(gè) servlet,則要使用 java.net.URLClassLoader 類,它是java.lang.ClassLoader 的間接子類。 一旦有了 URLClassLoader 類的實(shí)例,就可以使用 loadClass 方法來裝載一個(gè) servlet 類。 實(shí)例化 URLClassLoader 是很簡單的。 該類有三個(gè)構(gòu)建器,最簡單的是:?
public URLClassLoader(URL[] urls);?
urls 是一組指向其位置 java.net.URL 對象, 當(dāng)裝載一個(gè)類時(shí)它會(huì)自動(dòng)搜索其位置。任一以 / 結(jié)尾的 URL 都被假定為一目錄, 否則,就假定其為 .jar 文件,在需要時(shí)可以下載并打開。?
在一個(gè) servlet 容器內(nèi),類裝載器查找 servlet 類的位置稱為儲存庫 (repository)。在所舉的應(yīng)用程序中,類裝載器只可在當(dāng)前工作目錄下的 webroot/ 目錄查找,所以,首先得創(chuàng)建一組簡單的 URL。 URL 類提供了多個(gè)構(gòu)建器,因此有許多的方法來構(gòu)建一個(gè)URL 對象。 在這個(gè)應(yīng)用程序內(nèi),使用了和 TOMCAT 內(nèi)另外一個(gè)類所使用的相同的構(gòu)建器。 該構(gòu)建器頭部 (signature) 如下:?
public URL(URL context, String spec, URLStreamHandler hander)?
throws MalformedURLException?
可以通過傳遞給第二個(gè)參數(shù)一個(gè)規(guī)范,傳遞給第一個(gè)和第三個(gè)參數(shù) null 值來使用這個(gè)構(gòu)建器, 但在些有另外一種可接受三個(gè)參數(shù)的構(gòu)建器:?
public URL(String protocol, String host, String file)?
throws MalformedURLException?
因此,如果只寫了以下代碼,編譯器將不知道是使用的哪個(gè)構(gòu)建器:?
new URL(null, aString, null);?
當(dāng)然也可以能過告訴編譯器第三個(gè)參數(shù)的類型來避開這個(gè)問題,如:?
URLStreamHandler streamHandler = null;?
new URL(null, aString, streamHandler);?
對于第二個(gè)參數(shù),可以傳遞包含儲存庫 (repository) 的 String 。 以下代碼可創(chuàng)建:?
String repository = (new URL("file", null,?
classPath.getCanonicalPath() + File.separator)).toString();?
結(jié)合起來,以下是構(gòu)建正確 URLClassLoader 實(shí)例的 process 方法的部分代碼?
| // create a URLClassLoader URLStreamHandler streamHandler = null; URL[] urls = new URL[1]; File classPath = new File(Constants.WEB_ROOT); String repository = (new URL("file", null, classPath.getCanonicalPath() + File.separator)).toString() urls[0] = new URL(null, repository, streamHandler); loader = new URLClassLoader(urls); |
創(chuàng)建儲存庫 (repository)的代碼摘自org.apache.catalina.startup.ClassLoaderFactory內(nèi)的 createClassLoader 方法,而創(chuàng)建 URL 的代碼摘自org.apache.catalina.loader.StandardClassLoader 類內(nèi)的 addRepository 方法。 但在此階段您還沒有必要去關(guān)心這些類。?
有了類裝載器,您可以使用loadClass方法裝載servlet類:?
| Class myClass = null; try {myClass = loader.loadClass(servletName); } catch (ClassNotFoundException e) {System.out.println(e.toString()); } |
然后,process方法創(chuàng)建已裝載的 servlet類的實(shí)例,傳遞給 javax.servlet.Servlet ,并激活 servlet 的 service 方法:?
| Servlet servlet = null; try {servlet = (Servlet) myClass.newInstance();servlet.service((ServletRequest) request, (ServletResponse) response); } catch (Exception e) {System.out.println(e.toString()); } catch (Throwable e) {System.out.println(e.toString()); } |
編譯并運(yùn)行該應(yīng)用程序?
如果要編譯該應(yīng)用程序,在工作目錄下鍵入以下命令:?
javac -d . -classpath ./lib/servlet.jar src/ex02/pyrmont/*.java?
如果要在 windows 下運(yùn)行該應(yīng)用程序,在工作目錄下鍵入以下命令:?
java -classpath ./lib/servlet.jar;./ ex02.pyrmont.HttpServer1?
在 linux 環(huán)境下,使用冒號來隔開類庫:?
java -classpath ./lib/servlet.jar:./ ex02.pyrmont.HttpServer1?
如果要測試該應(yīng)用程序,請?jiān)?URL 或?yàn)g覽器地址欄鍵入以下命令:?
http://localhost:8080/index.html?
或者是:?
http://localhost:8080/servlet/PrimitiveServlet?
您將會(huì)在瀏覽器中看到以下文本:?
Hello. Roses are red.?
注意:您不能看到第二行字符 (Violets are blue),因?yàn)橹挥械谝恍凶址腿氲綖g覽器。 Tomcat 運(yùn)行工作原理 隨后的章節(jié)會(huì)告訴您怎樣來解決這個(gè)問題。
Servlet容器工作原理講解(4)
Application 2?
第一個(gè)應(yīng)用程序里存在一個(gè)值得注意的問題。 在ServletProcessor1 類的 process 方法里,上溯 (upcast)ex02.pyrmont.Request 實(shí)例到 javax.servlet.ServletRequest,將其作為第一個(gè)參數(shù)傳遞給 servlet 的 service 方法。 另上溯(upcast) ex02.pyrmont.Response 實(shí)例到 javax.servlet.ServletResponse ,并將其作為第二個(gè)參數(shù)傳遞給 servlet 的 service 方法。?
| try {servlet = (Servlet) myClass.newInstance();servlet.service((ServletRequest) request, (ServletResponse) response); } |
這樣會(huì)使安全性能大打折扣。 知道 servlet 容器工作原理的程序員可以將 ServletRequest 和 ServletResponse 實(shí)例向下轉(zhuǎn)型 (downcast) 到Request 和 Response ,并調(diào)用它們的 public 方法。 Request 實(shí)例能調(diào)用它的 parse 方法; Request 實(shí)例能調(diào)用它的 sendStaticResource 方法。?
可以將 parse 和 sendStaticResource 方法設(shè)為 private,因?yàn)樵?ex02.pyrmont 里將會(huì)從其他類里調(diào)用它們。 然而,這兩個(gè)方法在 servlet 內(nèi)應(yīng)該是不可用的。 一個(gè)解決方法是:給 Request 和 Response 類一個(gè)默認(rèn)的訪問修飾符,以致他們在 ex02.pyrmont 外不能被使用。 但還有一個(gè)更好的解決方法: 使用 facade 類。?
在第二個(gè)應(yīng)用程序內(nèi),添加兩個(gè) facade 類:RequestFacade 和 ResponseFacade。 RequestFacade 類實(shí)現(xiàn) ServletRequest 接口,并通過傳遞 Request 實(shí)例來實(shí)例化, Request 實(shí)例將在 ServletRequest 對象的構(gòu)建器里被引用 。 ServletRequest 對象本身是 private 類型的,不能在類之外訪問。 就構(gòu)建 RequestFacade 對象,并將其傳遞給 service 方法,而不上溯 (upcast) Request 對象給 ServletRequest,并將其傳遞給 service 方法。 servlet 程序員仍舊可以向下轉(zhuǎn)型 (downcast) ServletRequest 到 RequestFacade,但是,只要訪問 ServletRequest 接口的可用方法就可以了。 現(xiàn)在,parseUri 就安全了。?
Listing 2.5 顯示 RequestFacade 類部分代碼:?
Listing 2.5. RequestFacade 類?
| package ex02.pyrmont;public class RequestFacade implements ServletRequest {private ServletRequest request = null;public RequestFacade(Request request) {this.request = request;}/* implementation of the ServletRequest*/public Object getAttribute(String attribute) {return request.getAttribute(attribute);}public Enumeration getAttributeNames() {return request.getAttributeNames();}... } |
注意 RequestFacade 構(gòu)造函數(shù)。 它會(huì)接受一個(gè) Request 對象,即刻分配給私有的 servletRequest 對象引用。 還要注意,RequestFacade 內(nèi)的每個(gè)方法調(diào)用 ServletRequest 對象內(nèi)相應(yīng)的方法。?
ResponseFacade 類也是如此。?
以下是 application 2 所包含的類?
| HttpServer2 Request Response StaticResourceProcessor ServletProcessor2 Constants HttpServer2 類類似于 HttpServer1, 只是它在 await 方法內(nèi)使用了 ServletProcessor2 而不是ServletProcessor1。 if (request.getUri().startsWith("/servlet/")) {ServletProcessor2 processor = new ServletProcessor2();processor.process(request, response); } else {... } ServletProcessor2 類也類似于 ServletProcessor1, 只是在以下 process 方法的部分代碼有點(diǎn)不同: Servlet servlet = null; RequestFacade requestFacade = new RequestFacade(request); ResponseFacade responseFacade = new ResponseFacade(response);try {servlet = (Servlet) myClass.newInstance();servlet.service((ServletRequest) requestFacade, (ServletResponse) responseFacade); } |
編譯并運(yùn)行該應(yīng)用程序?
如果要編譯該應(yīng)用程序,在工作目錄下鍵入以下命令:?
javac -d . -classpath ./lib/servlet.jar src/ex02/pyrmont/*.java?
如果要在 windows 下運(yùn)行該應(yīng)用程序,在工作目錄下鍵入以下命令:?
java-classpath ./lib/servlet.jar;./ ex02.pyrmont.HttpServer2?
在linux環(huán)境下,使用分號來隔開類庫:?
java -classpath ./lib/servlet.jar:./ ex02.pyrmont.HttpServer2?
您可以使用和 application 1 相同的 URL 以收到同樣的結(jié)果。?
總結(jié)?
本文討論了簡單的能夠用于服務(wù)靜態(tài)資源,以及處理如 PrimitiveServlet 一樣簡單的 servlet 的 servlet 容器。 同時(shí)也提供 javax.servlet.Servlet 的背景信息。
總結(jié)
以上是生活随笔為你收集整理的servlet 工作原理的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 如何优雅的选择字体(font-famil
- 下一篇: Servlet3异步原理