本文源碼:GitHub·點這里 || GitEE·點這里
一、Servlet簡介
Java編寫的服務器端程序,具有獨立於平台和協議的特性,主要功能在於交互式地瀏覽和生成數據,生成動態Web內容。使用Servlet,可以收集來自網頁表單的用戶輸入,呈現來自數據庫或者其他源的記錄,還可以動態創建網頁。
二、實現方式
1、繼承HttpServlet
- API簡介
繼承自 GenericServlet. 遵守 HTTP協議實現,以設計模式的角度看,HttpServlet擔任抽象模板角色,模板方法:由service()方法擔任。基本方法:由doPost()、doGet()等方法擔任。service()方法流程,省略了部分判斷邏輯。該方法調用七個do方法中的一個或幾個,完成對客戶端請求的響應。這些do方法需要由HttpServlet的具體子類提供,這種API封裝是典型的模板方法模式。
- 代碼案例
public class ServletOneImpl extends HttpServlet {
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html;charset=utf-8");
response.getWriter().print("執行:doGet");
}
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html;charset=utf-8");
response.getWriter().print("執行:doPost");
}
}
2、繼承GenericServlet
- API 簡介
Servlet 接口和 ServletConfig 接口的實現類. 一個抽象類. 其中的 service 方法為抽象方法。
- 代碼案例
public class ServletTwoImpl extends GenericServlet {
@Override
public void service(ServletRequest servletRequest, ServletResponse servletResponse)
throws ServletException, IOException {
HttpServletResponse response = (HttpServletResponse)servletResponse ;
response.setContentType("text/html;charset=utf-8");
response.getWriter().print("執行:service");
}
}
3、實現Servlet接口
- API 簡介
Servlet是一個接口,其中包含init、getServletConfig、service、getServletInfo、destroy幾個核心方法。
- 代碼案例
public class ServletThreeImpl implements Servlet {
@Override
public void init(ServletConfig servletConfig) throws ServletException {
servletConfig.getServletName();
System.out.println("init 被調用...");
}
@Override
public void service(ServletRequest servletRequest, ServletResponse servletResponse)
throws ServletException, IOException {
System.out.println("ThreadId:"+Thread.currentThread().getId());
System.out.println("service 被調用...");
HttpServletResponse response = (HttpServletResponse)servletResponse ;
response.getWriter().print("Servlet.Life");
}
@Override
public void destroy() {
System.out.println("destroy 被調用...");
}
@Override
public ServletConfig getServletConfig() {
System.out.println("getServletConfig 被調用...");
return null;
}
@Override
public String getServletInfo() {
System.out.println("getServletInfo 被調用...");
return null;
}
}
三、生命周期
- 加載和實例化
當Servlet容器啟動或客戶端發送請求時,Servlet容器會查找是否存在該Servlet實例,若存在,則直接讀取該實例響應請求;如果不存在,就創建一個Servlet實例(屬於單例設計模式)。load-on-startup 可以配置創建時序。
- 初始化:init()
實例化后,Servlet容器將調用init方法一次,初始化當前 Servlet。
- 服務:service()
初始化后,Servlet處於響應請求的就緒狀態。當接收到客戶端請求時,調用service()的方法處理客戶端請求,HttpServlet的service()方法會根據不同的請求 調用不同的模板方法。
- 銷毀:destroy()
當Servlet容器關閉時,Servlet實例也隨時銷毀。關閉 Tomcat 服務時可以通過日志打印看到該方法的執行。
四、運行配置
1、web.xml配置
<servlet>
<servlet-name>servletOneImpl</servlet-name>
<servlet-class>com.node01.servlet.impl.ServletOneImpl</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>servletOneImpl</servlet-name>
<url-pattern>/servletOneImpl</url-pattern>
</servlet-mapping>
<servlet>
<servlet-name>servletTwoImpl</servlet-name>
<servlet-class>com.node01.servlet.impl.ServletTwoImpl</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>servletTwoImpl</servlet-name>
<url-pattern>/servletTwoImpl</url-pattern>
</servlet-mapping>
<servlet>
<servlet-name>servletThreeImpl</servlet-name>
<servlet-class>com.node01.servlet.impl.ServletThreeImpl</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>servletThreeImpl</servlet-name>
<url-pattern>/servletThreeImpl</url-pattern>
</servlet-mapping>
請求:http://localhost:6003/servletOneImpl
測試。
- servlet-name:Servlet 注冊名稱。
- servlet-class:Servlet 全路徑類名。
- serlvet-mapping:同一個Servlet可以被映射到多個URL上。
- url-pattern:Servlet 訪問的映射路徑。
2、線程池運行
觀察上述第三種Servlet實現方式的日志打印:Thread.currentThread().getId());
。
ThreadId:32
ThreadId:33
ThreadId:32
ThreadId:31
ThreadId:32
這里不難發現,Servlet以線程池的方式執行的。
五、源代碼地址
GitHub·地址
https://github.com/cicadasmile/java-base-parent
GitEE·地址
https://gitee.com/cicadasmile/java-base-parent