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()方法傳入瞭HttpServletRequestHttpServletResponse兩個對象,分別代表HTTP請求和響應。我們使用Servlet API時,並不直接與底層TCP交互,也不需要解析HTTP協議,因為HttpServletRequestHttpServletResponse就已經封裝好瞭請求和響應。以發送響應為例,我們隻需要設置正確的響應類型,然後獲取PrintWriter,寫入響應即可。

而這樣的一個項目最終會打包成一個*.war文件,運行這個文件,需要使用支持ServletAPI的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!

推薦閱讀: