Servlet生命周期與線程安全

来源:https://www.cnblogs.com/Y-oung/archive/2018/02/08/8433426.html
-Advertisement-
Play Games

上一篇介紹了Servlet初始化,以及如何處理HTTP請求,實際上在這兩個過程中,都伴隨著Servlet的生命周期,都是Servlet生命周期的一部分。同時,由於Tomcat容器預設是採用單實例多線程的方式處理多個請求,這一特性就導致了線程安全問題的存在。因此,本篇主要講述Servlet生命周期與線 ...


        上一篇介紹了Servlet初始化,以及如何處理HTTP請求,實際上在這兩個過程中,都伴隨著Servlet的生命周期,都是Servlet生命周期的一部分。同時,由於Tomcat容器預設是採用單實例多線程的方式處理多個請求,這一特性就導致了線程安全問題的存在。因此,本篇主要講述Servlet生命周期與線程安全問題。

        1、Servlet生命周期

        Servlet是運行在容器當中的,所以其生命周期也由容器控制,最常用的容器就是Tomcat,筆者經歷過的所有項目也都是以Tomcat作為Servlet的容器。經過前面幾篇介紹,相信大家對Servlet的生命周期有了一定的瞭解。Servlet的生命周期其實是通過javax.servlet.Servlet介面中的init()、service()、destroy()等方法表示的,主要有四個階段組成:載入並實例化、初始化(init())、處理請求(service())、銷毀(destroy())。下麵分別介紹這四個階段。

        載入並實例化

        Tomcat容器負責Servlet的載入並實例化,其實例化分兩種情況:當web.xml文件里配置了<load-on-startup>標簽並且裡面的數字>=0時,容器啟動時即載入Servlet類並創建類的實例;如果未配置<load-on-startup>標簽或數字<0時,容器啟動時不會載入Servlet類,當然也就不會創建類的實例。這時,當用戶首次訪問Servlet類時會載入並實例化。無論採用哪種方式實例化,都只會創建一個類的實例,無論多少用戶訪問Servlet,都共用這一個實例。

        初始化(init())

        在Servlet類實例化之後,容器將調用init()方法,傳遞ServletConfig介面的對象,進行初始化。在init()方法中,可以通過getServletConfig()方法獲取ServletConfig對象,然後通過此對象的getInitParameter()等方法獲取web.xml文件中<init-param>標簽裡面的配置信息,並對配置信息進行解析,或者執行任何其他一次性活動。在Servlet的整個生命周期中,init()方法只會被執行一次。

        處理請求(service())

        在Servlet初始化完成之後,容器就準備接收並處理客戶的請求了。處理請求時,容器會調用Servlet的service(HttpServletRequest req, HttpServletResponse resp)方法,這個方法會判斷用戶發送的請求類型,是“POST”請求還是“GET”請求或是其他請求,然後根據請求類型執行相應的doPost()方法、doGet()方法或其他方法。Tomcat容器會將用戶請求的數據封裝到HttpServletRequest對象中,伺服器處理完用戶請求之後,將結果信息返回到HttpServletResponse對象中,最終這兩個對象作為參數傳遞到doPost()、doGet()或其他方法中,將結果信息返回到頁面顯示。當多個客戶的請求到來時,伺服器會創建多個線程,每個客戶請求對應一個線程,每個請求的service()方法都能運行在自己獨立的線程中。

        銷毀(destroy())

        當Tomcat容器關閉時或由於其他原因導致Servlet需要關閉或卸載時,容器會調用該對象的destroy()方法,以便讓Servlet對象可以釋放它所使用的資源,該方法同樣只會執行一次。在容器調用destroy()方法前,如果還有其他的線程正在service()方法中執行,容器會等待這些線程執行完畢或者等待伺服器設定的超時值到達。一旦Servlet對象的destroy()方法被調用,容器會釋放這個Servlet對象,在隨後的時間內,該對象會被java的垃圾收集器所回收。這四個階段共同組成了Servlet的生命周期。

        2、Servlet線程安全

        通過上面的Servlet生命周期可以看出,在Tomcat容器載入並實例化Servlet之後,會創建一個實例,並且這個實例是唯一的,無論多少用戶訪問Servlet,都共用這一個實例。而每次用戶訪問Servlet時,伺服器都會為每個用戶創建一個獨立的線程,每個線程都有它自己的堆棧空間。所以說是單實例多線程,這種預設以多線程方式執行的設計可大大降低對系統的資源需求,提高系統的併發量及響應時間,但也同時引發了Servlet的線程安全問題。

        對於Servlet中的局部變數,多線程下每個線程對局部變數都會有自己的一份copy,存在自己的堆棧空間中,這樣對局部變數的修改只會影響到自己的copy而不會對別的線程產生影響,所以這是線程安全的;對於Servlet中的實例(全局)變數,多線程下所有線程共用實例變數,這一共用就可能導致多個線程之間互相影響,從而引發線程的不安全。

        知道了引發線程不安全問題的原因,那麼該如何預防這一情況發生呢?

        不使用實例變數

        既然實例變數能引發線程安全問題,那麼只要在Servlet類的任何方法裡面都不使用實例變數,該Servlet就是線程安全的。事實上,線程安全問題大部分是由實例變數造成的,在Servlet中避免使用實例變數是保證Servlet線程安全的最佳選擇。

        使用synchronized

        synchronized關鍵字能保證一次只有一個線程可以訪問被保護的區段,所以理論上可以通過同步塊操作來保證Servlet的線程安全。但因為其“一次只有一個線程可以訪問”的特性,導致當大量用戶訪問同一資源時,只能排隊訪問,大量用戶處於阻塞狀態,這就大大降低了其用戶的吞吐量,從而使系統的效率和性能大大降低,不推薦使用此方法。

        其他方式

        Java的有些集合類也會引發線程安全問題,應避免使用。比如用Vector代替ArrayList,用Hashtable代替HashMap等。另外,不要在Servlet中創建其他線程來完成某個功能,因為Servlet本身就是多線程的,再在Servlet中創建線程,更容易引發線程安全問題。

        轉載請註明出處 http://www.cnblogs.com/Y-oung/p/8433426.html

        工作、學習、交流或有任何疑問,請聯繫郵箱:[email protected]  微信:yy1340128046


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

-Advertisement-
Play Games
更多相關文章
  • #環境配置基於windows操作系統 (一) 環境配置 (1)執行下麵的命令(前提:已經安裝python環境,可以參考之前發的python筆記(一)) pip install -U selenium (2)http://docs.seleniumhq.org/download/ (網站打不開的話就翻 ...
  • 互斥鎖: 為什麼要有互斥鎖:由於多線程是並行的,如果某一線程取出了某一個數據將要進行操作,但它還沒有那麼快執行完操作,這時候如果另外一個線程也要操作這個數據,那麼這個數據可能會因為兩次操作而發生錯誤 import time,threading x=6 def run1(): print("run1我... ...
  • 本人錄製的Go入門視頻 "20小時快速入門go語言視頻" :https://pan.baidu.com/s/1jJPsThk 基礎編程 "01、Go語言介紹" "02、環境搭建" "03、第一個Go程式" "04、命名、變數、常量" "05、基礎數據類型" "06、格式化輸出、類型轉換、類型別名" ...
  • 為了方便,這次就不單獨寫腳本了,直接一步一步執行下來就好了先說下游標,就是一個指針,比如我有1234每條占一行,那麼初始游標預設是在1的位置,當read(1)後,游標自動向下next,現在指在2的位置,依次類推,然後是3,4直到最後,除非強制移動游標,否則游標不會再返回的今天寫了一個腳本,具體就是有 ...
  • 1、什麼是C+11 C++11標準為C++編程語言的第三個官方標準,正式名叫ISO/IEC 14882:2011 - Information technology -- Programming languages -- C++。在正式標準發佈前,原名C++0x。它將取代C++標準第二版ISO/IEC ...
  • 2018-02-08 23:32:23 修改context.xml文件 自從學習了servlet後,每次修改裡面的內容後,想要訪問都要重啟伺服器,這樣感覺很麻煩的,所以今天就教大家一個方法,只需要一行代碼就解決“無需重新啟動伺服器”。 請看下麵的圖示: 在第19行代碼處<Context>裡面寫上re ...
  • 使用內置函數 id( xx ) 獲取,返回結果以10進位格式表示 ...
  • hiernate4入門 【實戰】本文將講述hibernate4的入門,我寫是以實用入門為主,不作過多展開,我寫東西如果開頭標了實戰二字,必然會保證讀者能夠獲取源代碼並且能夠按要求做的話一定能跑起來,否則沒有意義。 【環境說明:】 jdk8,eclipse4.7,hibernate4.3.11,mys ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...