濮阳杆衣贸易有限公司

主頁(yè) > 知識(shí)庫(kù) > 分析Tomcat的工作原理

分析Tomcat的工作原理

熱門(mén)標(biāo)簽:云南云電銷(xiāo)機(jī)器人招商 400 電話 申請(qǐng)費(fèi)用 長(zhǎng)沙回?fù)芡夂粝到y(tǒng) ai電話機(jī)器人營(yíng)銷(xiāo) 江蘇自動(dòng)外呼系統(tǒng)一般多少錢(qián) 山東電信外呼系統(tǒng)靠譜嗎 鸚鵡螺號(hào)航海地圖標(biāo)注時(shí)間 比較穩(wěn)定的外呼系統(tǒng) 信貸電銷(xiāo)機(jī)器人系統(tǒng)

SpringBoot 就像一條巨蟒,慢慢纏繞著我們,使我們麻痹。不得不承認(rèn),使用了 SpringBoot 確實(shí)提高了工作效率,但同時(shí)也讓我們遺忘了很多技能。剛?cè)肷鐣?huì)的時(shí)候,我還是通過(guò) Tomcat 手動(dòng)部署 JavaWeb 項(xiàng)目,還經(jīng)常對(duì) Tomcat 進(jìn)行性能調(diào)優(yōu)。除此之外,還需要自己理清楚各 Jar 之間的關(guān)系,以避免 Jar 丟失和各版本沖突導(dǎo)致服務(wù)啟動(dòng)異常的問(wèn)題。到如今,這些繁瑣而又重復(fù)的工作已經(jīng)統(tǒng)統(tǒng)交給 SpringBoot 處理,我們可以把更多的精力放在業(yè)務(wù)邏輯上。但是,清楚 Tomcat 的工作原理和處理請(qǐng)求流程和分析 Spring 框架源碼一樣的重要。至少面試官特別喜歡問(wèn)這些底層原理和設(shè)計(jì)思路。希望這篇文章能給你一些幫助。

Tomcat 整體架構(gòu)

Tomcat 是一個(gè)免費(fèi)的、開(kāi)源的、輕量級(jí)的 Web 應(yīng)用服務(wù)器。適合在并發(fā)量不是很高的中小企業(yè)項(xiàng)目中使用。

文件目錄結(jié)構(gòu)

以下是 Tomcat 8 主要目錄結(jié)構(gòu)

目錄 功能說(shuō)明
bin 存放可執(zhí)行的文件,如 startup 和 shutdown
conf 存放配置文件,如核心配置文件 server.xml 和應(yīng)用默認(rèn)的部署描述文件 web.xml
lib 存放 Tomcat 運(yùn)行需要的jar包
logs 存放運(yùn)行的日志文件
webapps 存放默認(rèn)的 web 應(yīng)用部署目錄
work 存放 web 應(yīng)用代碼生成和編譯文件的臨時(shí)目錄

功能組件結(jié)構(gòu)

Tomcat 的核心功能有兩個(gè),分別是負(fù)責(zé)接收和反饋外部請(qǐng)求的連接器 Connector,和負(fù)責(zé)處理請(qǐng)求的容器 Container。其中連接器和容器相輔相成,一起構(gòu)成了基本的 web 服務(wù) Service。每個(gè) Tomcat 服務(wù)器可以管理多個(gè) Service。

組件 功能
Connector 負(fù)責(zé)對(duì)外接收反饋請(qǐng)求。它是 Tomcat 與外界的交通樞紐,監(jiān)聽(tīng)端口接收外界請(qǐng)求,并將請(qǐng)求處理后傳遞給容器做業(yè)務(wù)處理,最后將容器處理后的結(jié)果反饋給外界。
Container 負(fù)責(zé)對(duì)內(nèi)處理業(yè)務(wù)邏輯。其內(nèi)部由Engine、Host、Context 和 Wrapper 四個(gè)容器組成,用于管理和調(diào)用 Servlet 相關(guān)邏輯。
Service 對(duì)外提供的 Web 服務(wù)。主要包含連接器和容器兩個(gè)核心組件,以及其他功能組件。Tomcat 可以管理多個(gè) Service,且各 Service 之間相互獨(dú)立。

Tomcat 連接器核心原理

Tomcat 連接器框架——Coyote

連接器核心功能

一、監(jiān)聽(tīng)網(wǎng)絡(luò)端口,接收和響應(yīng)網(wǎng)絡(luò)請(qǐng)求。

二、網(wǎng)絡(luò)字節(jié)流處理。將收到的網(wǎng)絡(luò)字節(jié)流轉(zhuǎn)換成 Tomcat Request 再轉(zhuǎn)成標(biāo)準(zhǔn)的 ServletRequest 給容器,同時(shí)將容器傳來(lái)的 ServletResponse 轉(zhuǎn)成 Tomcat Response 再轉(zhuǎn)成網(wǎng)絡(luò)字節(jié)流。

連接器模塊設(shè)計(jì)

為滿(mǎn)足連接器的兩個(gè)核心功能,我們需要一個(gè)通訊端點(diǎn)來(lái)監(jiān)聽(tīng)端口;需要一個(gè)處理器來(lái)處理網(wǎng)絡(luò)字節(jié)流;最后還需要一個(gè)適配器將處理后的結(jié)果轉(zhuǎn)成容器需要的結(jié)構(gòu)。

組件 功能
Endpoint 端點(diǎn),用來(lái)處理 Socket 接收和發(fā)送的邏輯。其內(nèi)部由 Acceptor 監(jiān)聽(tīng)請(qǐng)求、Handler 處理數(shù)據(jù)、AsyncTimeout 檢查請(qǐng)求超時(shí)。具體的實(shí)現(xiàn)有 NioEndPoint、AprEndpoint 等。
Processor 處理器,負(fù)責(zé)構(gòu)建 Tomcat Request 和 Response 對(duì)象。具體的實(shí)現(xiàn)有 Http11Processor、StreamProcessor 等。
Adapter 適配器,實(shí)現(xiàn) Tomcat Request、Response 與 ServletRequest、ServletResponse之間的相互轉(zhuǎn)換。這采用的是經(jīng)典的適配器設(shè)計(jì)模式。
ProtocolHandler 協(xié)議處理器,將不同的協(xié)議和通訊方式組合封裝成對(duì)應(yīng)的協(xié)議處理器,如 Http11NioProtocol 封裝的是 HTTP + NIO。

對(duì)應(yīng)的源碼包路徑 org.apache.coyote 。對(duì)應(yīng)的結(jié)構(gòu)圖如下

Tomcat 容器核心原理

Tomcat 容器框架——Catalina

容器結(jié)構(gòu)分析

每個(gè) Service 會(huì)包含一個(gè)容器。容器由一個(gè)引擎可以管理多個(gè)虛擬主機(jī)。每個(gè)虛擬主機(jī)可以管理多個(gè) Web 應(yīng)用。每個(gè) Web 應(yīng)用會(huì)有多個(gè) Servlet 包裝器。Engine、Host、Context 和 Wrapper,四個(gè)容器之間屬于父子關(guān)系。

容器 功能
Engine 引擎,管理多個(gè)虛擬主機(jī)。
Host 虛擬主機(jī),負(fù)責(zé) Web 應(yīng)用的部署。
Context Web 應(yīng)用,包含多個(gè) Servlet 封裝器。
Wrapper 封裝器,容器的最底層。對(duì) Servlet 進(jìn)行封裝,負(fù)責(zé)實(shí)例的創(chuàng)建、執(zhí)行和銷(xiāo)毀功能。

對(duì)應(yīng)的源碼包路徑 org.apache.coyote 。對(duì)應(yīng)的結(jié)構(gòu)圖如下

容器請(qǐng)求處理

容器的請(qǐng)求處理過(guò)程就是在 Engine、Host、Context 和 Wrapper 這四個(gè)容器之間層層調(diào)用,最后在 Servlet 中執(zhí)行對(duì)應(yīng)的業(yè)務(wù)邏輯。各容器都會(huì)有一個(gè)通道 Pipeline,每個(gè)通道上都會(huì)有一個(gè) Basic Valve(如StandardEngineValve), 類(lèi)似一個(gè)閘門(mén)用來(lái)處理 Request 和 Response 。其流程圖如下。

Tomcat 請(qǐng)求處理流程

上面的知識(shí)點(diǎn)已經(jīng)零零碎碎地介紹了一個(gè) Tomcat 是如何處理一個(gè)請(qǐng)求。簡(jiǎn)單理解就是連接器的處理流程 + 容器的處理流程 = Tomcat 處理流程。哈!那么問(wèn)題來(lái)了,Tomcat 是如何通過(guò)請(qǐng)求路徑找到對(duì)應(yīng)的虛擬站點(diǎn)?是如何找到對(duì)應(yīng)的 Servlet 呢?

映射器功能介紹

這里需要引入一個(gè)上面沒(méi)有介紹的組件 Mapper。顧名思義,其作用是提供請(qǐng)求路徑的路由映射。根據(jù)請(qǐng)求URL地址匹配是由哪個(gè)容器來(lái)處理。其中每個(gè)容器都會(huì)它自己對(duì)應(yīng)的Mapper,如 MappedHost。不知道大家有沒(méi)有回憶起被 Mapper class not found 支配的恐懼。在以前,每寫(xiě)一個(gè)完整的功能,都需要在 web.xml 配置映射規(guī)則,當(dāng)文件越來(lái)越龐大的時(shí)候,各個(gè)問(wèn)題隨著也會(huì)出現(xiàn)

HTTP請(qǐng)求流程

打開(kāi) tomcat/conf 目錄下的 server.xml 文件來(lái)分析一個(gè)http://localhost:8080/docs/api 請(qǐng)求。

第一步:連接器監(jiān)聽(tīng)的端口是8080。由于請(qǐng)求的端口和監(jiān)聽(tīng)的端口一致,連接器接受了該請(qǐng)求。

第二步:因?yàn)橐娴哪J(rèn)虛擬主機(jī)是 localhost,并且虛擬主機(jī)的目錄是webapps。所以請(qǐng)求找到了 tomcat/webapps 目錄。

第三步:解析的 docs 是 web 程序的應(yīng)用名,也就是 context。此時(shí)請(qǐng)求繼續(xù)從 webapps 目錄下找 docs 目錄。有的時(shí)候我們也會(huì)把應(yīng)用名省略。

第四步:解析的 api 是具體的業(yè)務(wù)邏輯地址。此時(shí)需要從 docs/WEB-INF/web.xml 中找映射關(guān)系,最后調(diào)用具體的函數(shù)。

<?xml version="1.0" encoding="UTF-8"?>
<Server port="8005" shutdown="SHUTDOWN">

 <Service name="Catalina">

	<!-- 連接器監(jiān)聽(tīng)端口是 8080,默認(rèn)通訊協(xié)議是 HTTP/1.1 -->
 <Connector port="8080" protocol="HTTP/1.1"
  connectionTimeout="20000"
  redirectPort="8443" />
			 
	<!-- 名字為 Catalina 的引擎,其默認(rèn)的虛擬主機(jī)是 localhost -->
 <Engine name="Catalina" defaultHost="localhost">

	 <!-- 名字為 localhost 的虛擬主機(jī),其目錄是 webapps-->
 <Host name="localhost" appBase="webapps"
  unpackWARs="true" autoDeploy="true">

 </Host>
 </Engine>
 </Service>
</Server>

SpringBoot 如何啟動(dòng)內(nèi)嵌的 Tomcat

SpringBoot 一鍵啟動(dòng)服務(wù)的功能,讓有很多剛?cè)肷鐣?huì)的朋友都忘記 Tomcat 是啥。隨著硬件的性能越來(lái)越高,普通中小項(xiàng)目都可以直接用內(nèi)置 Tomcat 啟動(dòng)。但是有些大一點(diǎn)的項(xiàng)目可能會(huì)用到 Tomcat 集群和調(diào)優(yōu),內(nèi)置的 Tomcat 就不一定能滿(mǎn)足需求了。

我們先從源碼中分析 SpringBoot 是如何啟動(dòng) Tomcat,以下是 SpringBoot 2.x 的代碼。

代碼從 main 方法開(kāi)始,執(zhí)行 run 方法啟動(dòng)項(xiàng)目。

SpringApplication.run

從 run 方法點(diǎn)進(jìn)去,找到刷新應(yīng)用上下文的方法。

this.prepareContext(context, environment, listeners, applicationArguments, printedBanner);
this.refreshContext(context);
this.afterRefresh(context, applicationArguments);

從 refreshContext 方法點(diǎn)進(jìn)去,找 refresh 方法。并一層層往上找其父類(lèi)的方法。

this.refresh(context);

在 AbstractApplicationContext 類(lèi)的 refresh 方法中,有一行調(diào)用子容器刷新的邏輯。

this.postProcessBeanFactory(beanFactory);
this.invokeBeanFactoryPostProcessors(beanFactory);
this.registerBeanPostProcessors(beanFactory);
this.initMessageSource();
this.initApplicationEventMulticaster();
this.onRefresh();
this.registerListeners();
this.finishBeanFactoryInitialization(beanFactory);
this.finishRefresh();

從 onRefresh 方法點(diǎn)進(jìn)去,找到 ServletWebServerApplicationContext 的實(shí)現(xiàn)方法。在這里終于看到了希望。

protected void onRefresh() {
 super.onRefresh();

 try {
 this.createWebServer();
 } catch (Throwable var2) {
 throw new ApplicationContextException("Unable to start web server", var2);
 }
}

從 createWebServer 方法點(diǎn)進(jìn)去,找到從工廠類(lèi)中獲取 WebServer的代碼。

if (webServer == null && servletContext == null) {
 ServletWebServerFactory factory = this.getWebServerFactory();
 // 獲取 web server 
 this.webServer = factory.getWebServer(new ServletContextInitializer[]{this.getSelfInitializer()});
} else if (servletContext != null) {
 try {
 // 啟動(dòng) web server
 this.getSelfInitializer().onStartup(servletContext);
 } catch (ServletException var4) {
 throw new ApplicationContextException("Cannot initialize servlet context", var4);
 }
}

從 getWebServer 方法點(diǎn)進(jìn)去,找到 TomcatServletWebServerFactory 的實(shí)現(xiàn)方法,與之對(duì)應(yīng)的還有 Jetty 和 Undertow。這里配置了基本的連接器、引擎、虛擬站點(diǎn)等配置。

public WebServer getWebServer(ServletContextInitializer... initializers) {
 Tomcat tomcat = new Tomcat();
 File baseDir = this.baseDirectory != null ? this.baseDirectory : this.createTempDir("tomcat");
 tomcat.setBaseDir(baseDir.getAbsolutePath());
 Connector connector = new Connector(this.protocol);
 tomcat.getService().addConnector(connector);
 this.customizeConnector(connector);
 tomcat.setConnector(connector);
 tomcat.getHost().setAutoDeploy(false);
 this.configureEngine(tomcat.getEngine());
 Iterator var5 = this.additionalTomcatConnectors.iterator();

 while(var5.hasNext()) {
 Connector additionalConnector = (Connector)var5.next();
 tomcat.getService().addConnector(additionalConnector);
 }

 this.prepareContext(tomcat.getHost(), initializers);
 return this.getTomcatWebServer(tomcat);
}

服務(wù)啟動(dòng)后會(huì)打印日志

o.s.b.w.embedded.tomcat.TomcatWebServer : Tomcat initialized with port(s): 8900 (http)
o.apache.catalina.core.StandardService : Starting service [Tomcat]
org.apache.catalina.core.StandardEngine : Starting Servlet Engine: Apache Tomcat/8.5.34
o.a.catalina.core.AprLifecycleListener : The APR based Apache Tomcat Native library which allows optimal ...
o.a.c.c.C.[Tomcat].[localhost].[/] : Initializing Spring embedded WebApplicationContext
o.s.web.context.ContextLoader : Root WebApplicationContext: initialization completed in 16858 ms

END

文章到這里就結(jié)束了,實(shí)在是 hold 不住了,周末寫(xiě)了一整天還沒(méi)有寫(xiě)到源碼部分,只能放在下一章了。再寫(xiě)真的就要廢了,有什么不對(duì)的地方請(qǐng)多多指出

以上就是Tomcat的工作原理是怎樣的的詳細(xì)內(nèi)容,更多關(guān)于Tomcat 工作原理的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

標(biāo)簽:烏海 亳州 運(yùn)城 齊齊哈爾 嘉興 衡陽(yáng) 拉薩 澳門(mén)

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《分析Tomcat的工作原理》,本文關(guān)鍵詞  分析,Tomcat,的,工作,原理,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《分析Tomcat的工作原理》相關(guān)的同類(lèi)信息!
  • 本頁(yè)收集關(guān)于分析Tomcat的工作原理的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    盐城市| 彝良县| 天长市| 定结县| 宁德市| 深州市| 信丰县| 额尔古纳市| 拜泉县| 乌恰县| 夏河县| 筠连县| 镇平县| 阿合奇县| 吴川市| 南部县| 延寿县| 开封市| 昂仁县| 石阡县| 商南县| 阜新| 阿拉善左旗| 博爱县| 峨边| 锡林浩特市| 拜泉县| 敦煌市| 穆棱市| 菏泽市| 卓资县| 卫辉市| 西峡县| 泊头市| 西宁市| 海兴县| 黄冈市| 谢通门县| 冀州市| 霍邱县| 苗栗县|