Java中Servlet的生命周期詳解
Web基礎和HTTP協議
┌─────────┐ ┌─────────┐ │░░░░░░░░░│ │O ░░░░░░░│ ├─────────┤ ├─────────┤ │░░░░░░░░░│ │ │ ├─────────┤ │ │ │░░░░░░░░░│ └─────────┘ └─────────┘ │ request 1 │ │─────────────────────>│ │ request 2 │ │─────────────────────>│ │ response 1 │ │<─────────────────────│ │ request 3 │ │─────────────────────>│ │ response 3 │ │<─────────────────────│ │ response 2 │ │<─────────────────────│ ▼ ▼
我們註意到HTTP協議是一個請求-響應協議,它總是發送一個請求,然後接收一個響應。能不能一次性發送多個請求,然後再接收多個響應呢?HTTP 2.0
可以支持瀏覽器同時發出多個請求,但每個請求需要唯一標識,服務器可以不按請求的順序返回多個響應,由瀏覽器自己把收到的響應和請求對應起來。可見,HTTP 2.0
進一步提高瞭傳輸效率,因為瀏覽器發出一個請求後,不必等待響應,就可以繼續發下一個請求。
HTTP 3.0
為瞭進一步提高速度,將拋棄TCP
協議,改為使用無需創建連接的UDP
協議,目前HTTP 3.0
仍然處於實驗推廣階段。
什麼是Servlet
在JavaEE
平臺上,處理TCP
連接,解析HTTP
協議這些底層工作統統扔給現成的Web
服務器去做,我們隻需要把自己的應用程序跑在Web
服務器上。為瞭實現這一目的,JavaEE
提供瞭Servlet
API
,我們使用Servlet API
編寫自己的Servlet
來處理HTTP
請求,Web
服務器實現Servlet
API
接口,
實現底層功能.
// WebServlet註解表示這是一個Servlet,並映射到地址 hello.do @WebServlet(urlPatterns = "/hello.do") public class HelloServlet extends HttpServlet { protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { // 設置響應類型: resp.setContentType("text/html"); // 獲取輸出流: PrintWriter pw = resp.getWriter(); // 寫入響應: pw.write("<h1>Hello, world!</h1>"); // 最後不要忘記flush強制輸出: pw.flush(); } }
一個Servlet
總是繼承自HttpServlet
,然後覆寫doGet()
或doPost()
方法。註意到doGet()
方法傳入瞭HttpServletRequest
和HttpServletResponse
兩個對象,分別代表HTTP
請求和響應。我們使用Servlet API
時,並不直接與底層TCP
交互,也不需要解析HTTP
協議,因為HttpServletRequest
和HttpServletResponse
就已經封裝好瞭請求和響應。以發送響應為例,我們隻需要設置正確的響應類型,然後獲取PrintWriter
,寫入響應即可。
而這樣的一個項目最終會打包成一個*.war
文件,運行這個文件,需要使用支持Servlet
API
的Web容器(Web服務器)。
因此,我們首先要找一個支持Servlet API的Web服務器。
常用的服務器有:
- Tomcat:由Apache開發的開源免費服務器;
- Jetty:由Eclipse開發的開源免費服務器;
- GlassFish:一個開源的全功能JavaEE服務器。
Servlet的生命周期
在通過一個URL
路徑發起對一個Servlet
請求的過程中,其本質是在調用執行Servlet
實例的doXXX()
方法。該Servlet
實例創建和使用的過程,被稱為Servlet的生命周期。整個生命周期包括:實例化、初始化、服務、銷毀。
- 實例化:根據
Servlet
請求的路徑(例如:home.do
),查找該Servlet
的實例。如果實例不存在,則通過調用構造方法,完成Servlet
實例的創建。 - 初始化:通過該
Servlet
的實例,調用init()
方法,執行初始化的邏輯。 - 服務:通過該
Servlet
的實例,調用service()
方法,如果子類沒有重寫該方法,則調用HttpServlet父類的service()
方法,在父類的該方法中進行請求方式的判斷,如果是GET
請求,則調用doGet()
方法;如果是POST
請求,則調用doPost()
方法;
如果子類重寫doXXX()
方法,則調用子類重寫後的doXXX()
方法;
如果子類沒有重寫doXXX()
方法,則調用父類的doXXX()
方法,在父類的方法實現中,返回一個405
狀態碼的錯誤頁面。
405狀態碼:代表請求的方式服務器不提供支持。
4.銷毀:服務器關閉或重啟時,會銷毀所有的Servlet實例,會調用Servlet實例的destroy()
方法。
package com.my.hyz.web.servlet; import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; //@WebServlet("/home.do") public class HomeServlet extends HttpServlet { public HomeServlet() { System.out.println("實例化"); } @Override public void init() throws ServletException { System.out.println("初始化"); //super.init(); } @Override protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.println("調用Service實例"); } @Override public void destroy() { System.out.println("銷毀咯!!!!"); } @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { // TODO Auto-generated method stub System.out.println("哎呦get到瞭"+this.hashCode()); } @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.println("哎呦post到瞭"); } }
到此這篇關於Java中Servlet的生命周期詳解的文章就介紹到這瞭,更多相關Java Servlet 生命周期內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- 手工搭建Servlet實現
- SpringBoot中使用Servlet的兩種方式小結
- JavaWeb中請求轉發和請求重定向的區別以及使用
- 淺談Servlet的Cookie和Session機制
- 實現Servlet程序的三種方法(小結)