轉(zhuǎn):Tomcat啟動(dòng)過(guò)程原理詳解

原文地址:http://yut-i.blog.163.com/blog/static/2425778220121159347221/

基于Java的Web 應(yīng)用程序是 servlet知纷、JSP 頁(yè)面严衬、靜態(tài)頁(yè)面黄琼、類和其他資源的集合,它們可以用標(biāo)準(zhǔn)方式打包相寇,并運(yùn)行在來(lái)自多個(gè)供應(yīng)商的多個(gè)容器。Web 應(yīng)用程序存在于結(jié)構(gòu)化層次結(jié)構(gòu)的目錄中噪生,該層次結(jié)構(gòu)是由 Java Servlet 規(guī)范定義的裆赵。Web 應(yīng)用程序的根目錄包含直接存儲(chǔ)或存儲(chǔ)在子文件夾中的所有公共資源,比如圖像跺嗽、HTML 頁(yè)面等。構(gòu)成:Web應(yīng)用由Web組件(一組Java類庫(kù))页藻、html文件桨嫁,靜態(tài)資源文件(如圖像)、幫助類和庫(kù)組成份帐。

1 – Tomcat Server的組成部分
1.1 – Server

A Server element represents the entire Catalina servlet container. (Singleton)

1.2 – Service

A Service element represents the combination of one or more Connector components that share a single Engine

Service是這樣一個(gè)集合:它由一個(gè)或者多個(gè)Connector組成璃吧,以及一個(gè)Engine,負(fù)責(zé)處理所有Connector所獲得的客戶請(qǐng)求

1.3 – Connector

一個(gè)Connector將在某個(gè)指定端口上偵聽(tīng)客戶請(qǐng)求废境,并將獲得的請(qǐng)求交給Engine來(lái)處理畜挨,從Engine處獲得回應(yīng)并返回客戶。
TOMCAT有兩個(gè)典型的Connector噩凹,一個(gè)直接偵聽(tīng)來(lái)自browser的http請(qǐng)求巴元,一個(gè)偵聽(tīng)來(lái)自其它WebServer的請(qǐng)求。
Coyote Http/1.1 Connector 在端口8080處偵聽(tīng)來(lái)自客戶browser的http請(qǐng)求驮宴。
Coyote JK2 Connector 在端口8009處偵聽(tīng)來(lái)自其它WebServer(Apache)的servlet/jsp代理請(qǐng)求逮刨。

1.4 – Engine

The Engine element represents the entire request processing machinery associated with a particular Service. It receives and processes all requests from one or more Connectors and returns the completed response to the Connector for ultimate transmission back to the client

Engine下可以配置多個(gè)虛擬主機(jī)Virtual Host,每個(gè)虛擬主機(jī)都有一個(gè)域名堵泽,當(dāng)Engine獲得一個(gè)請(qǐng)求時(shí)修己,它把該請(qǐng)求匹配到某個(gè)Host上,然后把該請(qǐng)求交給該Host來(lái)處理迎罗,Engine有一個(gè)默認(rèn)虛擬主機(jī)睬愤,當(dāng)請(qǐng)求無(wú)法匹配到任何一個(gè)Host上的時(shí)候,將交給該默認(rèn)Host來(lái)處理纹安。

1.5 – Host

代表一個(gè)Virtual Host尤辱,虛擬主機(jī),每個(gè)虛擬主機(jī)和某個(gè)網(wǎng)絡(luò)域名Domain Name相匹配钻蔑。
每個(gè)虛擬主機(jī)下都可以部署(deploy)一個(gè)或者多個(gè)Web App啥刻,每個(gè)Web App對(duì)應(yīng)于一個(gè)Context,有一個(gè)Context path咪笑。
當(dāng)Host獲得一個(gè)請(qǐng)求時(shí)可帽,將把該請(qǐng)求匹配到某個(gè)Context上,然后把該請(qǐng)求交給該Context來(lái)處理
匹配的方法是“最長(zhǎng)匹配”窗怒,所以一個(gè)path==”"的Context將成為該Host的默認(rèn)Context映跟。
所有無(wú)法和其它Context的路徑名匹配的請(qǐng)求都將最終和該默認(rèn)Context匹配蓄拣。

1.6 – Context

一個(gè)Context對(duì)應(yīng)于一個(gè)Web Application,一個(gè)Web Application由一個(gè)或者多個(gè)Servlet組成努隙。
Context在創(chuàng)建的時(shí)候?qū)⒏鶕?jù)配置文件$CATALINA_HOME/conf/web.xml和$WEBAPP_HOME/WEB-INF/web.xml載入Servlet類球恤。
當(dāng)Context獲得請(qǐng)求時(shí),將在自己的映射表(mapping table)中尋找相匹配的Servlet類荸镊。
如果找到咽斧,則執(zhí)行該類,獲得請(qǐng)求的回應(yīng)躬存,并返回张惹。

2 – Tomcat Server的結(jié)構(gòu)圖
Tomcat Server結(jié)構(gòu)圖
3 – 配置文件$CATALINA_HOME/conf/server.xml的說(shuō)明

該文件描述了如何啟動(dòng)Tomcat Server

<!----------------------------------------------------------------------------------------------->
<!-- 啟動(dòng)Server 在端口8005處等待關(guān)閉命令 如果接受到"SHUTDOWN"字符串則關(guān)閉服務(wù)器 -->
<Server port="8005" shutdown="SHUTDOWN" debug="0">
<!-- Listener ??? 目前沒(méi)有看到這里 -->
<Listener className="org.apache.catalina.mbeans.ServerLifecycleListener" debug="0"/>
<Listener className="org.apache.catalina.mbeans.GlobalResourcesLifecycleListener" debug="0"/>
<!-- Global JNDI resources ??? 目前沒(méi)有看到這里,先略去 -->
<GlobalNamingResources>
... ... ... ...
</GlobalNamingResources>
<!-- Tomcat的Standalone Service Service是一組Connector的集合 它們共用一個(gè)Engine來(lái)處理所有Connector收到的請(qǐng)求 -->
<Service name="Tomcat-Standalone">
<!-- Coyote HTTP/1.1 Connector className : 該Connector的實(shí)現(xiàn)類是org.apache.coyote.tomcat4.CoyoteConnector port :
在端口號(hào)8080處偵聽(tīng)來(lái)自客戶browser的HTTP1.1請(qǐng)求 minProcessors : 該Connector先創(chuàng)建5個(gè)線程等待客戶請(qǐng)求岭洲,
每個(gè)請(qǐng)求由一個(gè)線程負(fù)責(zé) maxProcessors : 當(dāng)現(xiàn)有的線程不夠服務(wù)客戶請(qǐng)求時(shí)宛逗,若線程總數(shù)不足75個(gè),則創(chuàng)建新線程來(lái)處理請(qǐng)求
acceptCount : 當(dāng)現(xiàn)有線程已經(jīng)達(dá)到最大數(shù)75時(shí)盾剩,為客戶請(qǐng)求排隊(duì) 當(dāng)隊(duì)列中請(qǐng)求數(shù)超過(guò)100時(shí)雷激,后來(lái)的請(qǐng)求返回Connection refused
錯(cuò)誤 redirectport : 當(dāng)客戶請(qǐng)求是https時(shí),把該請(qǐng)求轉(zhuǎn)發(fā)到端口8443去 其它屬性略 -->
<Connector className="org.apache.coyote.tomcat4.CoyoteConnector"
port="8080"
minProcessors="5" maxProcessors="75" acceptCount="100"
enableLookups="true"
redirectPort="8443"
debug="0"
connectionTimeout="20000"
useURIValidationHack="false"
disableUploadTimeout="true" />
<!-- Engine用來(lái)處理Connector收到的Http請(qǐng)求 它將匹配請(qǐng)求和自己的虛擬主機(jī)告私,
并把請(qǐng)求轉(zhuǎn)交給對(duì)應(yīng)的Host來(lái)處理默認(rèn)虛擬主機(jī)是localhost -->
<Engine name="Standalone" defaultHost="localhost" debug="0">
<!-- 日志類屎暇,目前沒(méi)有看到,略去先 -->
<Logger className="org.apache.catalina.logger.FileLogger" .../>
<!-- Realm德挣,目前沒(méi)有看到恭垦,略去先 -->
<Realm className="org.apache.catalina.realm.UserDatabaseRealm" .../>
<!-- 虛擬主機(jī)localhost appBase : 該虛擬主機(jī)的根目錄是webapps/ 它將匹配請(qǐng)求和
自己的Context的路徑,并把請(qǐng)求轉(zhuǎn)交給對(duì)應(yīng)的Context來(lái)處理 -->
<Host name="localhost" debug="0" appBase="webapps" unpackWARs="true" autoDeploy="true">
<!-- 日志類格嗅,目前沒(méi)有看到番挺,略去先 -->
<Logger className="org.apache.catalina.logger.FileLogger" .../>
<!-- Context,對(duì)應(yīng)于一個(gè)Web App path : 該Context的路徑名是""屯掖,故該Context是該Host的
默認(rèn)Context docBase : 該Context的根目錄是webapps/mycontext/ -->
<Context path="" docBase="mycontext" debug="0"/>
<!-- 另外一個(gè)Context玄柏,路徑名是/wsota -->
<Context path="/wsota" docBase="wsotaProject" debug="0"/>
</Host>
</Engine>
</Service>
</Server>

<!----------------------------------------------------------------------------------------------->```
#####4 – Context的部署配置文件web.xml的說(shuō)明
一個(gè)Context對(duì)應(yīng)于一個(gè)Web App,每個(gè)Web App是由一個(gè)或者多個(gè)servlet組成的贴铜。
當(dāng)一個(gè)Web App被初始化的時(shí)候粪摘,它將用自己的ClassLoader對(duì)象載入“部署配置文件web.xml”中定義的每個(gè)servlet類筋遭。
它首先載入在$CATALINA_HOME/conf/web.xml中部署的servlet類姻氨。
然后載入在自己的Web App根目錄下的WEB-INF/web.xml中部署的servlet類篙贸。
web.xml文件有兩部分:servlet類定義和servlet映射定義谈宛。
每個(gè)被載入的servlet類都有一個(gè)名字,且被填入該Context的映射表(mapping table)中榛搔,和某種URL PATTERN對(duì)應(yīng)凌彬。
當(dāng)該Context獲得請(qǐng)求時(shí)挖藏,將查詢mapping table,找到被請(qǐng)求的servlet勤讽,并執(zhí)行以獲得請(qǐng)求回應(yīng)蟋座。

分析一下所有的Context共享的web.xml文件,在其中定義的servlet被所有的Web App載入

<web-app>

<servlet>

<servlet-name>default</servlet-name>

<servlet-class>

org.apache.catalina.servlets.DefaultServlet

</servlet-class>

<init-param>

<param-name>debug</param-name>

<param-value>0</param-value>

</init-param>

<init-param>

<param-name>listings</param-name>

<param-value>true</param-value>

</init-param>

<load-on-startup>1</load-on-startup>

</servlet>

<servlet>

<servlet-name>invoker</servlet-name>

<servlet-class>org.apache.catalina.servlets.InvokerServlet </servlet-class>

<init-param>

 <param-name>debug</param-name>

 <param-value>0</param-value>

</init-param>

<load-on-startup>2</load-on-startup>

</servlet>

<servlet>

<servlet-name>jsp</servlet-name>

<servlet-class>org.apache.jasper.servlet.JspServlet</servlet-class>

<init-param>

 <param-name>logVerbosityLevel</param-name>

 <param-value>WARNING</param-value>

</init-param>

<load-on-startup>3</load-on-startup>

</servlet>

<servlet-mapping>

<servlet-name>default</servlet-name>

<url-pattern>/</url-pattern>

</servlet-mapping>

<servlet-mapping>

<servlet-name>invoker</servlet-name>

<url-pattern>/servlet/*</url-pattern>

</servlet-mapping>

<servlet-mapping>

<servlet-name>jsp</servlet-name>

<url-pattern>*.jsp</url-pattern>

</servlet-mapping>

... ... ... ...

</web-app>

```

5 – Tomcat Server處理一個(gè)http請(qǐng)求的過(guò)程

假設(shè)來(lái)自客戶的請(qǐng)求為:

http://localhost:8080/wsota/wsota_index.jsp

  1. 請(qǐng)求被發(fā)送到本機(jī)端口8080脚牍,被在那里偵聽(tīng)的Coyote HTTP/1.1 Connector獲得
  2. Connector把該請(qǐng)求交給它所在的Service的Engine來(lái)處理向臀,并等待來(lái)自Engine的回應(yīng)
  3. Engine獲得請(qǐng)求localhost/wsota/wsota_index.jsp,匹配它所擁有的所有虛擬主機(jī)Host
  4. Engine匹配到名為localhost的Host(即使匹配不到也把請(qǐng)求交給該Host處理诸狭,因?yàn)樵揌ost被定義為該Engine的默認(rèn)主機(jī))
  5. localhost Host獲得請(qǐng)求/wsota/wsota_index.jsp券膀,匹配它所擁有的所有Context
  6. Host匹配到路徑為/wsota的Context(如果匹配不到就把該請(qǐng)求交給路徑名為”"的Context去處理)
  7. path=”/wsota”的Context獲得請(qǐng)求/wsota_index.jsp,在它的mapping table中尋找對(duì)應(yīng)的servlet
  8. Context匹配到URL PATTERN為*.jsp的servlet驯遇,對(duì)應(yīng)于JspServlet類
  9. 構(gòu)造HttpServletRequest對(duì)象和HttpServletResponse對(duì)象三娩,作為參數(shù)調(diào)用JspServlet的doGet或doPost方法
    10)Context把執(zhí)行完了之后的HttpServletResponse對(duì)象返回給Host
    11)Host把HttpServletResponse對(duì)象返回給Engine
    12)Engine把HttpServletResponse對(duì)象返回給Connector
    13)Connector把HttpServletResponse對(duì)象返回給客戶browser
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市妹懒,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌双吆,老刑警劉巖眨唬,帶你破解...
    沈念sama閱讀 217,509評(píng)論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異好乐,居然都是意外死亡匾竿,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,806評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門蔚万,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)岭妖,“玉大人,你說(shuō)我怎么就攤上這事反璃£腔牛” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 163,875評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵淮蜈,是天一觀的道長(zhǎng)斋攀。 經(jīng)常有香客問(wèn)我,道長(zhǎng)梧田,這世上最難降的妖魔是什么淳蔼? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,441評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮裁眯,結(jié)果婚禮上鹉梨,老公的妹妹穿的比我還像新娘。我一直安慰自己穿稳,他們只是感情好存皂,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,488評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著司草,像睡著了一般艰垂。 火紅的嫁衣襯著肌膚如雪泡仗。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,365評(píng)論 1 302
  • 那天猜憎,我揣著相機(jī)與錄音娩怎,去河邊找鬼。 笑死胰柑,一個(gè)胖子當(dāng)著我的面吹牛截亦,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播柬讨,決...
    沈念sama閱讀 40,190評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼崩瓤,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了踩官?” 一聲冷哼從身側(cè)響起却桶,我...
    開(kāi)封第一講書(shū)人閱讀 39,062評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎蔗牡,沒(méi)想到半個(gè)月后颖系,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,500評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡辩越,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,706評(píng)論 3 335
  • 正文 我和宋清朗相戀三年嘁扼,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片黔攒。...
    茶點(diǎn)故事閱讀 39,834評(píng)論 1 347
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡趁啸,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出督惰,到底是詐尸還是另有隱情不傅,我是刑警寧澤,帶...
    沈念sama閱讀 35,559評(píng)論 5 345
  • 正文 年R本政府宣布姑丑,位于F島的核電站蛤签,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏栅哀。R本人自食惡果不足惜震肮,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,167評(píng)論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望留拾。 院中可真熱鬧戳晌,春花似錦、人聲如沸痴柔。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,779評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至豪嚎,卻和暖如春搔驼,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背侈询。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,912評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工舌涨, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人扔字。 一個(gè)月前我還...
    沈念sama閱讀 47,958評(píng)論 2 370
  • 正文 我出身青樓囊嘉,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親革为。 傳聞我的和親對(duì)象是個(gè)殘疾皇子扭粱,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,779評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • 轉(zhuǎn)自陳明乾的博客,可能有一定更新震檩。 轉(zhuǎn)原文聲明:原創(chuàng)作品琢蛤,允許轉(zhuǎn)載,轉(zhuǎn)載時(shí)請(qǐng)務(wù)必以超鏈接形式標(biāo)明文章 原始出處 抛虏、...
    C86guli閱讀 4,684評(píng)論 6 72
  • 0 系列目錄# WEB請(qǐng)求處理 WEB請(qǐng)求處理一:瀏覽器請(qǐng)求發(fā)起處理 WEB請(qǐng)求處理二:Nginx請(qǐng)求反向代理 本...
    七寸知架構(gòu)閱讀 13,957評(píng)論 22 190
  • 前言 Tomcat隸屬于Apache基金會(huì)虐块,是開(kāi)源的輕量級(jí)Web應(yīng)用服務(wù)器,使用非常廣泛嘉蕾。server.xml是T...
    余平的余_余平的平閱讀 1,935評(píng)論 0 23
  • 太陽(yáng)總算出來(lái)了,出門不做蒙頭鳥(niǎo)霜旧。脫掉羽絨服错忱,單衣遮嫩肉。 街頭銀發(fā)聚挂据,巷尾游公主以清。田野菜花黃,蜜蜂雙腿忙崎逃。 ...
    木貞ma閱讀 56評(píng)論 0 1
  • 讀書(shū)掷倔,記讀書(shū)筆記,寫(xiě)讀后感个绍,記錄點(diǎn)滴勒葱。 不斷輸入,學(xué)以致用巴柿,學(xué)以編碼 “分享"快樂(lè)凛虽,“分享”成長(zhǎng),所以分享我的筆記...
    hxleihao閱讀 139評(píng)論 0 0