濮阳杆衣贸易有限公司

主頁 > 知識庫 > 淺談Tomcat多層容器的設(shè)計

淺談Tomcat多層容器的設(shè)計

熱門標(biāo)簽:藍點外呼系統(tǒng) 貴陽教育行業(yè)電話外呼系統(tǒng) 威海人工外呼系統(tǒng)供應(yīng)商 烏海智能電話機器人 撫順移動400電話申請 在百度地圖標(biāo)注車輛 400電話申請方案 寧夏房產(chǎn)智能外呼系統(tǒng)要多少錢 做外呼系統(tǒng)的公司違法嗎

Tomcat的容器用來裝載Servlet。那Tomcat的Servlet容器是如何設(shè)計的呢?

容器的層次結(jié)構(gòu)

Tomcat設(shè)計了4種容器:Engine、Host、Context和Wrapper

Tomcat通過這種分層,使得Servlet容器具有很好的靈活性。

  • Context表示一個Web應(yīng)用程序
  • Wrapper表示一個Servlet,一個Web應(yīng)用程序中可能會有多個Servlet
  • Host代表一個虛擬主機,或一個站點,可以給Tomcat配置多個虛擬主機地址,而一個虛擬主機下可以部署多個Web應(yīng)用程序
  • Engine表示引擎,用來管理多個虛擬站點,一個Service最多只能有一個Engine

觀察Tomcat的server.xml配置文件。Tomcat采用了組件化設(shè)計,最外層即是Server

這些容器具有父子關(guān)系,形成一個樹形結(jié)構(gòu),Tomcat用組合模式來管理這些容器。

所有容器組件都實現(xiàn)Container接口,因此組合模式可以使得用戶對

單容器對象
最底層的Wrapper

組合容器對象
上面的Context、Host或者Engine
的使用具有一致性。

Container接口定義:

public interface Container extends Lifecycle {
    public void setName(String name);
    public Container getParent();
    public void setParent(Container container);
    public void addChild(Container child);
    public void removeChild(Container child);
    public Container findChild(String name);
}

請求定位Servlet的過程

搞這么多層次的容器,Tomcat是怎么確定請求是由哪個Wrapper容器里的Servlet來處理的呢?
Tomcat用Mapper組件完成這個任務(wù)。

Mapper就是將用戶請求的URL定位到一個Servlet

工作原理

Mapper組件保存了Web應(yīng)用的配置信息:容器組件與訪問路徑的映射關(guān)系,比如

  • Host容器里配置的域名
  • Context容器里的Web應(yīng)用路徑
  • Wrapper容器里Servlet映射的路徑

這些配置信息就是一個多層次的Map。

當(dāng)一個請求到來時,Mapper組件通過解析請求URL里的域名和路徑,再到自己保存的Map里去查找,就能定位到一個Servlet。
一個請求URL最后只會定位到一個Wrapper容器,即一個Servlet。

假如有一網(wǎng)購系統(tǒng),有

  • 面向B端管理人員的后臺管理系統(tǒng)
  • 面向C端用戶的在線購物系統(tǒng)

這倆系統(tǒng)跑在同一Tomcat,為隔離它們的訪問域名,配置兩個虛擬域名:

manage.shopping.com
管理人員通過該域名訪問Tomcat去管理用戶和商品,而用戶管理和商品管理是兩個單獨的Web應(yīng)用

user.shopping.com
C端用戶通過該域名去搜索商品和下訂單,搜索功能和訂單管理也是兩個獨立Web應(yīng)用

這樣部署,Tomcat會創(chuàng)建一個Service組件和一個Engine容器組件,在Engine容器下創(chuàng)建兩個Host子容器,在每個Host容器下創(chuàng)建兩個Context子容器。由于一個Web應(yīng)用通常有多個Servlet,Tomcat還會在每個Context容器里創(chuàng)建多個Wrapper子容器。每個容器都有對應(yīng)訪問路徑

Tomcat如何將URL定位到一個Servlet呢?

首先,根據(jù)協(xié)議和端口號選定Service和Engine
Tomcat的每個連接器都監(jiān)聽不同的端口,比如Tomcat默認的HTTP連接器監(jiān)聽8080端口、默認的AJP連接器監(jiān)聽8009端口。該URL訪問8080端口,因此會被HTTP連接器接收,而一個連接器是屬于一個Service組件的,這樣Service組件就確定了。一個Service組件里除了有多個連接器,還有一個Engine容器,因此Service確定了,Engine也確定了。

根據(jù)域名選定Host。
Mapper組件通過URL中的域名去查找相應(yīng)的Host容器,比如user.shopping.com,因此Mapper找到Host2容器。

根據(jù)URL路徑找到Context組件
Host確定以后,Mapper根據(jù)URL的路徑來匹配相應(yīng)的Web應(yīng)用的路徑,比如例子中訪問的是/order,因此找到了Context4這個Context容器。

最后,根據(jù)URL路徑找到Wrapper(Servlet)
Context確定后,Mapper再根據(jù)web.xml中配置的Servlet映射路徑來找到具體Wrapper和Servlet。

并非只有Servlet才會去處理請求,查找路徑上的父子容器都會對請求做一些處理:

  • 連接器中的Adapter會調(diào)用容器的Service方法執(zhí)行Servlet
  • 最先拿到請求的是Engine容器,Engine容器對請求做一些處理后,會把請求傳給自己子容器Host繼續(xù)處理,依次類推
  • 最后這個請求會傳給Wrapper容器,Wrapper會調(diào)用最終的Servlet來處理

這個調(diào)用過程使用的Pipeline-Valve管道,責(zé)任鏈模式,在一個請求處理的過程中有很多處理者依次對請求進行處理,每個處理者負責(zé)做自己相應(yīng)的處理,處理完之后將再調(diào)用下一個處理者繼續(xù)處理。

Valve表示一個處理點,比如權(quán)限認證和記錄日志。

public interface Valve {
  public Valve getNext();
  public void setNext(Valve valve);
  public void invoke(Request request, Response response)
}

由于Valve是一個處理點,因此invoke方法就是來處理請求的。
Pipeline接口:

public interface Pipeline extends Contained {
  public void addValve(Valve valve);
  public Valve getBasic();
  public void setBasic(Valve valve);
  public Valve getFirst();
}

所以Pipeline中維護了Valve鏈表,Valve可插入到Pipeline。
Pipeline中沒有invoke方法,因為整個調(diào)用鏈的觸發(fā)是Valve完成自己的處理后,調(diào)用getNext.invoke調(diào)用下一個Valve。

每個容器都有一個Pipeline對象,只要觸發(fā)這個Pipeline的第一個Valve,這個容器里Pipeline中的Valve就都會被調(diào)用到。但不同容器的Pipeline如何鏈?zhǔn)接|發(fā)?
比如Engine中Pipeline需要調(diào)用下層容器Host中的Pipeline。
Pipeline有個getBasic方法。這個BasicValve處于Valve鏈尾,負責(zé)調(diào)用下層容器的Pipeline里的第一個Valve。


整個調(diào)用過程由連接器中的Adapter觸發(fā)的,它會調(diào)用Engine的第一個Valve:

Wrapper

容器的最后一個Valve會創(chuàng)建一個Filter鏈,并調(diào)用doFilter方法,最終會調(diào)到Servlet的service方法。

Valve和Filter有什么區(qū)別呢?

  • Valve是Tomcat的私有機制,與Tomcat緊耦合。Servlet API是公有標(biāo)準(zhǔn),所有Web容器包括Jetty都支持Filter
  • Valve工作在Web容器級別,攔截所有應(yīng)用的請求。Servlet Filter工作在應(yīng)用級別,只攔截某個Web應(yīng)用的所有請求。若想做整個Web容器的攔截器,必須使用Valve。

到此這篇關(guān)于淺談Tomcat多層容器的設(shè)計的文章就介紹到這了,更多相關(guān)Tomcat 多層容器內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

標(biāo)簽:周口 松原 那曲 蕪湖 慶陽 泰州 銅川 朝陽

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《淺談Tomcat多層容器的設(shè)計》,本文關(guān)鍵詞  淺談,Tomcat,多層,容器,的,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《淺談Tomcat多層容器的設(shè)計》相關(guān)的同類信息!
  • 本頁收集關(guān)于淺談Tomcat多層容器的設(shè)計的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    永安市| 东乌珠穆沁旗| 福泉市| 克什克腾旗| 湘潭县| 乡城县| 札达县| 阿克苏市| 朝阳县| 巴彦淖尔市| 宜丰县| 雅安市| 东安县| 秦皇岛市| 镇坪县| 辉南县| 吉木乃县| 鸡泽县| 灵山县| 贵州省| 格尔木市| 西藏| 山东| 赞皇县| 金溪县| 宝丰县| 辉县市| 光山县| 米易县| 赤壁市| 兴宁市| 嘉黎县| 上饶市| 巴彦淖尔市| 逊克县| 应用必备| 大名县| 云南省| 平顺县| 大田县| 平远县|