如何證明Servlet是單例的?

来源:https://www.cnblogs.com/skyblue-li/archive/2023/05/25/17431463.html
-Advertisement-
Play Games

Servlet是web體系裡面最重要的部分,下麵羅列幾道常見的面試題,小伙伴們一定要好好記住哈。 1.Servlet是單例的嗎,如何證明? Servlet一般都是單例的,並且是多線程的。如何證明Servlet是單例模式呢?很簡單,重寫Servlet的init方法,或者添加一個構造方法。然後,在web ...


Servlet是web體系裡面最重要的部分,下麵羅列幾道常見的面試題,小伙伴們一定要好好記住哈。

1.Servlet是單例的嗎,如何證明?

Servlet一般都是單例的,並且是多線程的。如何證明Servlet是單例模式呢?很簡單,重寫Servlet的init方法,或者添加一個構造方法。然後,在web.xml中配置。如:

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://xmlns.jcp.org/xml/ns/javaee" xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd" id="WebApp_ID" version="3.1">
  

  <servlet>
    <servlet-name>MyServlet</servlet-name>
    <servlet-class>web.MyServlet</servlet-class>
  </servlet>
  <servlet-mapping>
    <servlet-name>MyServlet</servlet-name>
    <url-pattern>/hello</url-pattern>
  </servlet-mapping>

</web-app>

然後是MyServlet

public class MyServlet extends HttpServlet{
 
 public MyServlet(){
  System.out.println("MyServlet構造函數調用了");
 }

 @Override
 public void init() throws ServletException {
  System.out.println("MyServlet初始化");
 }
 
 

}

啟動Tomcat,不管你訪問多少次這個Servlet,init方法和構造器都只會執行1次。

2.如何讓Servlet變成多例

方法1.實現 SingleThreadModel 介面(不推薦,官方已經將這個介面廢棄)

public class MyServlet extends HttpServlet implements SingleThreadModel{
 
 public MyServlet(){
  System.out.println("MyServlet構造函數調用了");
 }

 @Override
 public void init() throws ServletException {
  System.out.println("MyServlet初始化");
 }

}

SingleThreadModel的意思是“單線程模式”,如果servlet實現了該介面,會確保不會有兩個線程同時執行servlet的service方法。

servlet容器通過同步化訪問servlet的單實例來保證,也可以通過維持servlet的實例池,對於新的請求會分配給一個空閑的servlet。源碼中,最多會生成20個實例。

方法2. 在web.xml中多配置一個Servlet

哪怕是同一個Servlet,你在web.xml中配置幾個,就會有幾個實例。

3.你能證明Servlet線程不安全嗎?

Servlet預設是線程不安全的!

Servlet體繫結構是建立在Java多線程機制之上的,它的生命周期是由Web容器負責的。

當客戶端第一次請求某個Servlet時,Servlet容器將會根據web.xml配置文件實例化這個Servlet類。

當有新的客戶端請求該Servlet時,一般不會再實例化該Servlet類,也就是有多個線程在使用這個實例。

Servlet容器會自動使用線程池等技術來支持系統的運行。

當兩個或多個線程同時訪問同一個Servlet時,可能會發生多個線程同時訪問同一資源的情況,數據可能會變得不一致。

所以在用Servlet構建的Web應用時如果不註意線程安全的問題,會使所寫的Servlet程式有難以發現的錯誤。

下麵舉一個例子來說明,為什麼Servlet是線程不安全的。

public class MyServlet extends HttpServlet{
 
 String message;

 @Override
 protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
  message = req.getParameter("message");
  PrintWriter out = resp.getWriter();
  //故意延時5秒鐘,使得下一次請求過來的時候,message的值還沒有返回就被覆蓋了
  try {
   Thread.sleep(5000);
  } catch (InterruptedException e) {
   e.printStackTrace();
  }
  out.write(message);
  out.flush();
  out.close();
  
 }


}

打開兩個瀏覽器,分別訪問:

http://localhost:8080/web/hello?message=jack

http://localhost:8080/web/hello?message=rose

因為有5秒的延時,所以可能就會出現第一個Servlet還沒返回呢,第二個Servlet就進來了。於是,把message的值給衝掉了。如下圖

石錘了,Servlet是線程不安全的。

4.你怎麼設計一個線程安全的Servlet?

1.最直接的辦法,就是用上面的SingleThreadModel介面

既然單例會有共用實例變數導致線程不安全的問題,那就改成多例的唄。

但是,這個介面都已經被官方廢棄了,這就說明官方也不推薦這麼做。原因很簡單,那就是這樣一來會有很多個實例,性能的代價太大了。

  1. 用同步鎖

這也是非常容易想到的辦法,把當前對象鎖起來,不返回不給其他用戶插入(怎麼有點怪怪的?)

@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
 
 synchronized(this){
  message = req.getParameter("message");
  PrintWriter out = resp.getWriter();
  //故意延時5秒鐘,使得下一次請求過來的時候,message的值還沒有釋放
  try {
   Thread.sleep(5000);
  } catch (InterruptedException e) {
   e.printStackTrace();
  }
  out.write(message);
  out.flush();
  out.close();
 
 }
 
}

這樣的代價就是等待時間更長了,參考火車上的的衛生間,這就是同步鎖。

  1. 儘量別用實例變數,用局部變數代替

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 本篇帶你走進AIGC的基本使用,一步一步註冊ChatGPT,申請自己的API進行使用,解決代理的問題,最後介紹如何本地部署ChatGPT,以及通過免費雲平臺搭建代理轉發,從而不需要使用魔法就可以訪問。 ...
  • JVM(Java虛擬機)是Java程式的運行環境,它可以通過一些系統參數進行配置和優化。以下是一些常用的JVM系統參數: 1. -Xmx: 用於設置JVM堆的最大記憶體大小。例如,-Xmx1g表示將堆的最大大小設置為1GB。 2. -Xms: 用於設置JVM堆的初始記憶體大小。例如,-Xms512m表示 ...
  • 基本數據類型和字元串類型的自動轉換<%@ taglib prefix="form" uri="http://www.springframework.org/tags/form" %> <%@ page contentType="text/html;charset=UTF-8" language="j ...
  • [toc] 你好!我是[@馬哥python說](https://www.zhihu.com/people/13273183132),一名10年程式猿,正在試錯用pyecharts開發可視化大屏的非常規排版。 以下,我用8種ThemeType展示的同一個可視化數據大屏,可視化主題是分析**“淄博燒烤” ...
  • 來源:https://www.duidaima.com/Group/Topic/JAVA/11942 ## **1、什麼是狀態機** ### 1.1 什麼是狀態 先來解釋什麼是“狀態”( State )。現實事物是有不同狀態的,例如一個自動門,就有 open 和 closed 兩種狀態。我們通常所說 ...
  • [toc] # 高階函數 高階函數是將函數用作參數或返回值的函數,還可以把函數賦值給一個變數。 所有函數類型都有一個圓括弧括起來的參數類型列表以及一個返回類型:(A, B) -> C 表示接受類型分別為 A 與 B 兩個參數並返回一個 C 類型值的函數類型。 參數類型列表可以為空,如 () -> A ...
  • 本文將為大家詳細講解Java中的Map集合,這是我們進行開發時經常用到的知識點,也是大家在學習Java中很重要的一個知識點,更是我們在面試時有可能會問到的問題。文章較長,乾貨滿滿,建議大家收藏慢慢學習。文末有本文重點總結,主頁有全系列文章分享。技術類問題,歡迎大家和我們一起交流討論! ...
  • # 0.相關確定 本教程使用的版本號為專業版PyCharm 2022.3.2,如果您是初學者,為了更好的學習本教程,避免不必要的麻煩,請您下載使用與本教程一致的版本號。 # 1.PyCharm的下載 官網下載:https://www.jetbrains.com/pycharm/download/ot ...
一周排行
    -Advertisement-
    Play Games
  • 在一些複雜的業務表中間查詢數據,有時候操作會比較複雜一些,不過基於SqlSugar的相關操作,處理的代碼會比較簡單一些,以前我在隨筆《基於SqlSugar的開發框架循序漸進介紹(2)-- 基於中間表的查詢處理》介紹過基於主表和中間表的聯合查詢,而往往實際會比這個會複雜一些。本篇隨筆介紹聯合多個表進行... ...
  • 從按鈕、文本框到下拉框、列表框,WPF提供了一系列常用控制項,每個控制項都有自己獨特的特性和用途。通過靈活的佈局容器,如網格、堆棧面板和換行面板,我們可以將這些控制項組合在一起,實現複雜的界面佈局。而通過樣式和模板,我們可以輕鬆地定製控制項的外觀和行為,以符合我們的設計需求。本篇記錄WPF入門需要瞭解的樣式... ...
  • 以MySQL資料庫為例 # 一. 安裝 NuGet搜索Dapper.Lite並安裝最新版本。 ![](https://img2023.cnblogs.com/blog/174862/202306/174862-20230602155913303-757935399.jpg) NuGet搜索MySql ...
  • # 圖片介面JWT鑒權實現 # 前言 之前做了個返回圖片鏈接的介面,然後沒做授權,然後今天鍵盤到了,也是用JWT來做介面的許可權控制。 然後JTW網上已經有很多文章來說怎麼用了,這裡就不做多的解釋了,如果不懂的可以參考下列鏈接的 文章。 圖片介面文章:[還在愁個人博客沒有圖片放?](https://w ...
  • ![線程各屬性縱覽](https://img2023.cnblogs.com/blog/1220983/202306/1220983-20230603114109107-477345835.png) 如上圖所示,線程有四個屬性: - 線程ID - 線程名稱 - 守護線程 - 線程優先順序 ### 1. ...
  • 本次主要介紹golang中的標準庫`bytes`,基本上參考了 [位元組 | bytes](https://cloud.tencent.com/developer/section/1140520) 、[Golang標準庫——bytes](https://www.jianshu.com/p/e6f7f2 ...
  • 歡迎來到本篇文章!通過上一篇什麼是 Spring?為什麼學它?的學習,我們知道了 Spring 的基本概念,知道什麼是 Spring,以及為什麼學習 Spring。今天,這篇就來說說 Spring 中的核心概念之一 IoC。 ...
  • # 2022版本IDEA+Maven+Tomcat的第一個程式(傻瓜教學) ​ 作為學習Javaweb的一個重要環節,如何實現在IDEA中利用Maven工具創建一個Javaweb程式模版並連接Tomcat發佈是非常重要的。我比較愚鈍(小白),而且自身電腦先前運行過spring或maven的程式,系統 ...
  • 本篇專門扯一下有關 QCheckBox 組件的一個問題。老周不水字數,直接上程式,你看了就明白。 #include <QApplication> #include <QWidget> #include <QPushButton> #include <QCheckBox> #include <QVBo ...
  • # 1.列表數據元素排序 在創建的列表中,數據元素的排列順序常常是無法預測的。這雖然在大多數情況下都是不可避免的,但經常需要以特定的順序呈現信息。有時候希望保留列表數據元素最初的排列順序,而有時候又需要調整排列順序。python提供了很多列表數據元素排序的方式,可根據情況選用。 ## 1.永久性排序 ...