tomcat伺服器源碼解讀01-整體結構

来源:https://www.cnblogs.com/wangrq/archive/2020/06/28/tomcat-01.html
-Advertisement-
Play Games

tomcat伺服器源碼解讀,整體結構梳理,開源server,java servlet容器 ...


tomcat介紹

開源的 Java Web 應用伺服器,實現了Java EE(Java Platform Enterprise Edition)的部分技術規範,比如 Java Servlet、JavaServer Pages、Java Expression Language、Java WebSocket。

(接觸最多的一款開源伺服器,研究下源碼結構,可加深對BS請求過程的理解)

目錄結構

bin目錄: 主要是用來存放tomcat的腳本,如startup.sh , shutdown.sh

conf 目錄: 下是配置文件

  • catalina.policy: Tomcat安全策略文件,控制JVM相關許可權,具體可以參考java. security.Permission

  • catalina.properties : Tomcat Catalina行為控制配置文件,比如Common ClassLoader

  • logging.properties : Tomcat日誌配置文件, JDK Logging

  • server.xml : Tomcat Server配置文件

  • GlobalNamingResources :全局JNDI資源

  • context.xml : 全局Context配置文件

  • tomcat-users.xml : Tomcat角色配置文件

  • web.xml : Servlet標準的web.xml部署文件, Tomcat預設實現部分配置入內:

    • org.apache.catalina.servlets.DefaultServlet

    • org.apache.jasper.servlet.JspServlet

lib目錄: 公共類庫

logs目錄: tomcat在運行過程中產生的日誌文件

webapps: 用來存放應用程式,當tomcat啟動時會去載入webapps目錄下的應用程式

work: 用來存放tomcat在運行時的編譯後文件,例如JSP編譯後的文件

配置文件及腳本

# /bin/startup.sh 啟動
EXECUTABLE=catalina.sh
exec "$PRGDIR"/"$EXECUTABLE" start "$@"
# /bin/shutdown.sh 關閉
EXECUTABLE=catalina.sh
exec "$PRGDIR"/"$EXECUTABLE" stop "$@"
# 啟動和關閉都是調用 catalina.sh 腳本
# /bin/catalina.sh 發現如下2行
org.apache.catalina.startup.Bootstrap "$@" start
org.apache.catalina.startup.Bootstrap "$@" stop

org.apache.catalina.startup.Bootstrap類是入口類,內部含有main方法,可以以此查看源碼

#catalina.properties
#限制可以訪問的包
package.access=sun.,org.apache.catalina.,org.apache.coyote.,org.apache.jasper.,org.apache.tomcat.
#common類載入器可以載入的lib資源,catalina.base與catalina.home是相同
common.loader="${catalina.base}/lib","${catalina.base}/lib/*.jar","${catalina.home}/lib","${catalina.home}/lib/*.jar"
server.loader=預設空,公用common.loader
shared.loader=預設空,公用common.loader
<!-- server.xml  -->
<Server port="8005" shutdown="SHUTDOWN">
  <Service name="Catalina">
    <Executor name="tomcatThreadPool"namePrefix="exec-my"prestartminSpareThreads="true"
              maxThread="200"maxThreads="500" minSpareThreads="8"maxIdleTime="10000"/>
    <Connector port="8080"protocol="HTTP/1.1"executor="tomcatThreadPool"connectionTimeout="20000"redirectPort="8443" />
    <Engine name="Catalina" defaultHost="localhost">
      <Host name="localhost" appBase="webapps" unpackWARs="true" autoDeploy="true">
        <!--<Context docBase="D:\myapp" path="/xxx"  reloadable="true" />-->
      </Host>
    </Engine>
  </Service>
</Server>

web應用部署

1、部署到webapps目錄下,該目錄下預設每個目錄都是一個應用,可以在server.xml文件中用 <Host/>標簽自定義目錄位置

<Host name="localhost" appBase="webapps" unpackWARs="true" autoDeploy="true">

2、在server.xml文件中配置Context標簽

<Context docBase="D:\myapp" path="/xxx"  reloadable="true" />

 path: 指定訪問該Web應用的URL入口

 docBase: 指定Web應用的文件路徑,可以給定絕對路徑,也可以給定相對於<Host>的appBase屬性的相對路徑。

 reloadable: 如果這個屬性設為true,tomcat伺服器在運行狀態下會監視在WEB-INF/classes和WEB-INF/lib目錄下class文件的改動,

 如果監測到有class文件被更新的,伺服器會自動重新載入Web應用。

3、獨立的Context xml文件配置

      在$CATALINA_BASE/conf/[enginename]/[hostname]/ 目錄下(預設conf/Catalina/localhost)創建xml文件,文件名就是contextPath

      比如創建api.xml,path就是/api, 註意:想要根目錄訪問,文件名為ROOT.xml

官方架構圖

 

 

啟動過程

org.apache.catalina.startup.Bootstrap#main()-> bootstrap.init(); daemon.load(args); daemon.start();
org.apache.catalina.startup.Catalina#load()-> digester.parse(server.xml); getServer().init(); # start()->getServer().start();
org.apache.catalina.core.StandardServer#startInternal()-> for:services[i].start(); initInternal():for:services[i].init();
org.apache.catalina.core.StandardService#startInternal() -> engine.start(); for:executor.start(); for:connector.start();
org.apache.catalina.core.StandardEngine#startInternal()->
           findChildren().for:executor.submit(new StartChild(children[i]));->FutureTask->StartChild.start()
           ((Lifecycle) pipeline).start();
           threadStart()->new Thread(new ContainerBackgroundProcessor()).start();

核心組件

1、Server (org.apache.catalina.Server)

      是指整個 Tomcat 伺服器,包含多組服務,負責管理和啟動各個Service,同時監聽 8005 埠發過來的 shutdown 命令,用於關閉整個容器; org.apache.catalina.core.StandardServer

      

 

 2、Service (org.apache.catalina.Service)

       Tomcat封裝的、對外提供完整的基於組件的web服務,含有Connectors,Container2個核心組件,以及多個功能組件,各個service之間是獨立的,共用同一個JVM資源,每個service組件都包含了若幹個用於接收客戶端消息的connector組件和處理請求的Engine組件.

       service組件還包含若幹個Executor組件,每個都是一個線程池,他可以為service內所有組件提供線程池執行任務. org.apache.catalina.core.StandardService。

       

 

  3、Connector

       Tomcat 與外部世界的連接器,監聽固定埠接收外部請求,傳遞給 Container,並將Container 處理的結果返回給外部.

org.apache.coyote.http11.Http11AprProtocol  // AprEndpoint
org.apache.coyote.http11.Http11NioProtocol  // NioEndpoint
org.apache.coyote.http11.Http11Nio2Protocol // Nio2Endpoint

 

 4、Container

       Catalina,Servlet容器,內部有多層容器組成,用於管理 Servlet 生命周期,調用 servlet 相關方法。

  • Engine : Servlet 的頂層容器,包含一個或多個 Host 子容器;

  • Host:虛擬主機,負責 web 應用的部署和 Context 的創建;

  • Context:Web 應用上下文,包含多個 Wrapper,負責 web 配置的解析、管理所有的 Web 資源;

  • Wrapper:最底層的容器,是對 Servlet 的封裝,負責 Servlet 實例的創建、執行和銷毀。

      

// 子容器啟動過程
org.apache.catalina.core.ContainerBase#startStopExecutor.submit(new StartChild(children[i]))
FutureTask->StartChild.start()

Context 應用載入

  tomcat是如何載入web項目

  • WEB-INF/web.xml

  • 零xml配置

// spi @HandlesTypes(WebApplicationInitializer.class)
org.apache.catalina.core.StandardContext#startInternal() org.springframework.web.SpringServletContainerInitializer#onStartup() ContextConfig#webConfig() org.apache.catalina.startup.ContextConfig#configureContext()
//<load-on-startup>1</load-on-startup> org.apache.catalina.core.StandardContext#loadOnStartup() //servlet 初始化

Tomcat啟動機制(外置和內嵌)

1、Tomcat啟動帶動IoC容器啟動的邏輯

     

 

     Spring boot中Tomcat容器和IoC容器的啟動順序

  • war外置: Tomcat啟動帶動IoC容器啟動

  • 內嵌: Ioc容器帶動Tomcat啟動

org.springframework.boot.web.embedded.tomcat.TomcatWebServer#start
org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory#getWebServer
org.springframework.context.support.AbstractApplicationContext#refresh

其他組件

  • Loader:封裝了 Java ClassLoader,用於 Container 載入類文件;

  • Session:負責管理和創建 session,以及 Session 的持久化(可自定義),支持 session 的集群。

  • Pipeline:在容器中充當管道的作用,管道中可以設置各種 valve(閥門),請求和響應在經由管道中各個閥門處理,提供了一種靈活可配置的處理請求和響應的機制。

  • JMX:Java SE 中定義技術規範,是一個為應用程式、設備、系統等植入管理功能的框架,通過 JMX 可以遠程監控 Tomcat 的運行狀態;

  • Realm:Tomcat 中為 web 應用程式提供訪問認證和角色管理的機制;

  • Jasper:Tomcat 的 Jsp 解析引擎,用於將 Jsp 轉換成 Java 文件,並編譯成 class 文件。

  • Naming:命名服務,JNDI, Java 命名和目錄介面,是一組在 Java 應用中訪問命名和目錄服務的 API。命名服務將名稱和對象聯繫起來,使得我們可以用名稱訪問對象,目錄服務也是一種命名 服務,對象不但有名稱,還有屬性。Tomcat 中可以使用 JNDI 定義數據源、配置信息,用於開發 與部署的分離。

官網連接  


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

-Advertisement-
Play Games
更多相關文章
  • Web 創建設計 設計一個網站,需要認真思考和規劃。 最重要的是要知道你的訪問用戶。 用戶是瀏覽者 一個典型的訪問者將無法讀取您的網頁的全部內容! 無論您在網頁中發佈了多麼有用的信息,一個訪問者在決定是否繼續閱讀之前僅僅會花幾秒鐘的時間進行瀏覽。 請確保使你的觀點,在頁面的第一句!另外,您還需要在整 ...
  • powertools可以稱得上插件界的瑞士軍刀。 相對於VS Code中大多數插件的出現為瞭解決某一項弊端和不足,powertools則聚合了很多強大且實用的功能,能夠增強VS Code的功能,並提升VS Code的使用體驗。 powertools就如同之前使用iOS系統時使用過的一款軟體Workf ...
  • CommonJS 和 ES6 Module 究竟有什麼區別? 作為前端開發者,你是否也曾有過疑惑,為什麼可以代碼中可以直接使用 require 方法載入模塊,為什麼載入第三方包的時候 Node 會知道選擇哪個文件作為入口,以及常被問到的,為什麼 ES6 Module export 基礎數據類型的時候 ...
  • 在 Node.js 項目開發過程中,隨著項目的發展,調用關係越來越複雜,調試工具的重要性日益凸顯。 Node(v6.3+)集成了方便好用 V8 Inspect 調試器,允許我們通過 Chrome DevTools 進行圖形化的調試和性能分析。同時,我們也可以使用 VS Code,Webstorm 等 ...
  • 歡迎使用慕課網 - Markdown 編輯器 Markdown 編輯器使用一套簡單實用的標記語言來實現簡單的文本排版,可以讓你專註於鍵盤碼字而非排版,化繁為簡,回歸寫作本質,帶來前所未有的書寫體驗! 我們在工具欄提供了豐富的快捷鍵,可以使用它們標記不同的標題,將一些文字標記為粗體或者斜體,也可以創建 ...
  • /** * 多個關鍵詞列表高亮(word_list1,color1,word_list2,color2,...) * @param word_list 關鍵詞列表(例: ["關鍵詞a","關鍵詞b"],不區分大小寫) * @param color 顏色值(例: "#ff0000") * @retur ...
  • 普利姆演算法(加點法)求最小生成樹 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> ...
  • jdk1.7中的底層實現過程(底層基於數組+鏈表) 在我們new HashMap()時,底層創建了預設長度為16的一維數組Entry[ ] table。當我們調用map.put(key1,value1)方法向HashMap里添加數據的時候: 首先,調用key1所在類的hashCode()計算key1 ...
一周排行
    -Advertisement-
    Play Games
  • 基於.NET Framework 4.8 開發的深度學習模型部署測試平臺,提供了YOLO框架的主流系列模型,包括YOLOv8~v9,以及其系列下的Det、Seg、Pose、Obb、Cls等應用場景,同時支持圖像與視頻檢測。模型部署引擎使用的是OpenVINO™、TensorRT、ONNX runti... ...
  • 十年沉澱,重啟開發之路 十年前,我沉浸在開發的海洋中,每日與代碼為伍,與演算法共舞。那時的我,滿懷激情,對技術的追求近乎狂熱。然而,隨著歲月的流逝,生活的忙碌逐漸占據了我的大部分時間,讓我無暇顧及技術的沉澱與積累。 十年間,我經歷了職業生涯的起伏和變遷。從初出茅廬的菜鳥到逐漸嶄露頭角的開發者,我見證了 ...
  • C# 是一種簡單、現代、面向對象和類型安全的編程語言。.NET 是由 Microsoft 創建的開發平臺,平臺包含了語言規範、工具、運行,支持開發各種應用,如Web、移動、桌面等。.NET框架有多個實現,如.NET Framework、.NET Core(及後續的.NET 5+版本),以及社區版本M... ...
  • 前言 本文介紹瞭如何使用三菱提供的MX Component插件實現對三菱PLC軟元件數據的讀寫,記錄了使用電腦模擬,模擬PLC,直至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1. PLC開發編程環境GX Works2,GX Works2下載鏈接 https:// ...
  • 前言 整理這個官方翻譯的系列,原因是網上大部分的 tomcat 版本比較舊,此版本為 v11 最新的版本。 開源項目 從零手寫實現 tomcat minicat 別稱【嗅虎】心有猛虎,輕嗅薔薇。 系列文章 web server apache tomcat11-01-官方文檔入門介紹 web serv ...
  • 1、jQuery介紹 jQuery是什麼 jQuery是一個快速、簡潔的JavaScript框架,是繼Prototype之後又一個優秀的JavaScript代碼庫(或JavaScript框架)。jQuery設計的宗旨是“write Less,Do More”,即倡導寫更少的代碼,做更多的事情。它封裝 ...
  • 前言 之前的文章把js引擎(aardio封裝庫) 微軟開源的js引擎(ChakraCore))寫好了,這篇文章整點js代碼來測一下bug。測試網站:https://fanyi.youdao.com/index.html#/ 逆向思路 逆向思路可以看有道翻譯js逆向(MD5加密,AES加密)附完整源碼 ...
  • 引言 現代的操作系統(Windows,Linux,Mac OS)等都可以同時打開多個軟體(任務),這些軟體在我們的感知上是同時運行的,例如我們可以一邊瀏覽網頁,一邊聽音樂。而CPU執行代碼同一時間只能執行一條,但即使我們的電腦是單核CPU也可以同時運行多個任務,如下圖所示,這是因為我們的 CPU 的 ...
  • 掌握使用Python進行文本英文統計的基本方法,並瞭解如何進一步優化和擴展這些方法,以應對更複雜的文本分析任務。 ...
  • 背景 Redis多數據源常見的場景: 分區數據處理:當數據量增長時,單個Redis實例可能無法處理所有的數據。通過使用多個Redis數據源,可以將數據分區存儲在不同的實例中,使得數據處理更加高效。 多租戶應用程式:對於多租戶應用程式,每個租戶可以擁有自己的Redis數據源,以確保數據隔離和安全性。 ...