不少開源應用服務器都是集成tomcat做爲web container的,並且對於tomcat的servlet container這部分代碼不多改動。這樣,這些應用服務器的性能基本上就取決於Tomcat處理HTTP請求的connector模塊的性能。本文首先從應用層次分析了tomcat全部的connector種類及用法,接着從架構上分析了connector模塊在整個tomcat中所處的位置,最後對connector作了詳細的源代碼分析。而且咱們以Http11NioProtocol爲例詳細說明了tomcat是如何經過實現ProtocolHandler接口而構建connector的。html
經過本文的學習,應該能夠輕鬆作到將tomcat作爲web container集成到第三方系統,而且自定義任何你想要的高性能的HTTP鏈接器。java
1 Connector介紹
1.1 Connector的種類
Tomcat源碼中與connector相關的類位於org.apache.coyote包中,Connector分爲如下幾類:web
- Http Connector, 基於HTTP協議,負責創建HTTP鏈接。它又分爲BIO Http Connector與NIO Http Connector兩種,後者提供非阻塞IO與長鏈接Comet支持。
- AJP Connector, 基於AJP協議,AJP是專門設計用來爲tomcat與http服務器之間通訊專門定製的協議,能提供較高的通訊速度和效率。如與Apache服務器集成時,採用這個協議。
- APR HTTP Connector, 用C實現,經過JNI調用的。主要提高對靜態資源(如HTML、圖片、CSS、JS等)的訪問性能。如今這個庫已獨立出來可用在任何項目中。Tomcat在配置APR以後性能很是強勁。
1.2 Connector的配置
對Connector的配置位於conf/server.xml文件中。apache
1.2.1 BIO HTTP/1.1 Connector配置
一個典型的配置以下:設計模式
<Connector port=」8080」 protocol=」HTTP/1.1」 maxThreads=」150」
connectionTimeout=」20000」 redirectPort=」8443」
其它一些重要屬性以下:瀏覽器
- acceptCount : 接受鏈接request的最大鏈接數目,默認值是10
- address : 綁定IP地址,若是不綁定,默認將綁定任何IP地址
- allowTrace : 若是是true,將容許TRACE HTTP方法
- compressibleMimeTypes : 各個mimeType, 以逗號分隔,如text/html,text/xml
- compression : 若是帶寬有限的話,能夠用GZIP壓縮
- connectionTimeout : 超時時間,默認爲60000ms (60s)
- maxKeepAliveRequest : 默認值是100
- maxThreads : 處理請求的Connector的線程數目,默認值爲200
若是是SSL配置,以下:tomcat
<Connector port="8181" protocol="HTTP/1.1" SSLEnabled="true"
maxThreads="150" scheme="https" secure="true"
clientAuth="false" sslProtocol = "TLS"
address="0.0.0.0"
keystoreFile="E:/java/jonas-full-5.1.0-RC3/conf/keystore.jks"
keystorePass="changeit" />
其中,keystoreFile爲證書位置,keystorePass爲證書密碼服務器
1.2.2 NIO HTTP/1.1 Connector配置
<Connector port=」8080」 protocol=」org.apache.coyote.http11.Http11NioProtocol」
maxThreads=」150」 connectionTimeout=」20000」 redirectPort=」8443」
1.2.3 Native APR Connector配置
- ARP是用C/C++寫的,對靜態資源(HTML,圖片等)進行了優化。因此要下載本地庫
tcnative-1.dll與openssl.exe,將其放在%tomcat%\bin目錄下。架構
下載地址是:http://tomcat.heanet.ie/native/1.1.10/binaries/win32/app
- 在server.xml中要配置一個Listener,以下圖。這個配置tomcat是默認配好的。
<!--APR library loader. Documentation at /docs/apr.html -->
<Listener className="org.apache.catalina.core.AprLifecycleListener" SSLEngine="on" />
- 配置使用APR connector
<Connector port=」8080」 protocol=」org.apache.coyote.http11.Http11AprProtocol」 maxThreads=」150」 connectionTimeout=」20000」 redirectPort=」8443」
- 若是配置成功,啓動tomcat,會看到以下信息:
org.apache.coyote.http11.Http11AprProtocol init
2 Connector在Tomcat中所處的位置
2.1 Tomcat架構
圖2-1 Tomcat架構
- Server(服務器)是Tomcat構成的頂級構成元素,全部一切均包含在Server中,Server的實現類StandardServer能夠包含一個到多個Services;
- 次頂級元素Service的實現類爲StandardService調用了容器(Container)接口,實際上是調用了Servlet Engine(引擎),並且StandardService類中也指明瞭該Service歸屬的Server;
- 接下來次級的構成元素就是容器(Container),主機(Host)、上下文(Context)和引擎(Engine)均繼承自Container接口,因此它們都是容器。可是,它們是有父子關係的,在主機(Host)、上下文(Context)和引擎(Engine)這三類容器中,引擎是頂級容器,直接包含是主機容器,而主機容器又包含上下文容器,因此引擎、主機和上下文從大小上來講又構成父子關係,雖然它們都繼承自Container接口。
- 鏈接器(Connector)將Service和Container鏈接起來,首先它須要註冊到一個Service,它的做用就是把來自客戶端的請求轉發到Container(容器),這就是它爲何稱做鏈接器的緣由。
故咱們從功能的角度將Tomcat源代碼分紅5個子模塊,它們分別是:
- Jsper子模塊:這個子模塊負責jsp頁面的解析、jsp屬性的驗證,同時也負責將jsp頁面動態轉換爲java代碼並編譯成class文件。在Tomcat源代碼中,凡是屬於org.apache.jasper包及其子包中的源代碼都屬於這個子模塊;
- Servlet和Jsp規範的實現模塊:這個子模塊的源代碼屬於javax.servlet包及其子包,如咱們很是熟悉的javax.servlet.Servlet接口、javax.servet.http.HttpServlet類及javax.servlet.jsp.HttpJspPage就位於這個子模塊中;
- Catalina子模塊:這個子模塊包含了全部以org.apache.catalina開頭的java源代碼。該子模塊的任務是規範了Tomcat的整體架構,定義了Server、Service、Host、Connector、Context、Session及Cluster等關鍵組件及這些組件的實現,這個子模塊大量運用了Composite設計模式。同時也規範了Catalina的啓動及中止等事件的執行流程。從代碼閱讀的角度看,這個子模塊應該是咱們閱讀和學習的重點。
- Connectors子模塊:若是說上面三個子模塊實現了Tomcat應用服務器的話,那麼這個子模塊就是Web服務器的實現。所謂鏈接器(Connector)就是一個鏈接客戶和應用服務器的橋樑,它接收用戶的請求,並把用戶請求包裝成標準的Http請求(包含協議名稱,請求頭Head,請求方法是Get仍是Post等等)。同時,這個子模塊還按照標準的Http協議,負責給客戶端發送響應頁面,好比在請求頁面未發現時,connector就會給客戶端瀏覽器發送標準的Http 404錯誤響應頁面。
- Resource子模塊:這個子模塊包含一些資源文件,如Server.xml及Web.xml配置文件。嚴格說來,這個子模塊不包含java源代碼,可是它仍是Tomcat編譯運行所必需的。
2.2 Tomcat運行流程
圖2-2 tomcat運行流程
假設來自客戶的請求爲:http://localhost:8080/test/index.jsp
- 請求被髮送到本機端口8080,被在那裏偵聽的Coyote HTTP/1.1 Connector得到
- Connector把該請求交給它所在的Service的Engine來處理,並等待Engine的迴應
- Engine得到請求localhost:8080/test/index.jsp,匹配它全部虛擬主機Host
- Engine匹配到名爲localhost的Host(即便匹配不到也把請求交給該Host處理,由於該Host被定義爲該Engine的默認主機)
- localhost Host得到請求/test/index.jsp,匹配它所擁有的全部Context
- Host匹配到路徑爲/test的Context(若是匹配不到就把該請求交給路徑名爲""的Context去處理)
- path="/test"的Context得到請求/index.jsp,在它的mapping table中尋找對應的servlet
- Context匹配到URL PATTERN爲*.jsp的servlet,對應於JspServlet類
- 構造HttpServletRequest對象和HttpServletResponse對象,做爲參數調用JspServlet的doGet或doPost方法
- Context把執行完了以後的HttpServletResponse對象返回給Host
- Host把HttpServletResponse對象返回給Engine
- Engine把HttpServletResponse對象返回給Connector
- Connector把HttpServletResponse對象返回給客戶browser
3 Connector源碼分析
3.1 Tomcat的啓動分析與集成設想
咱們知道,啓動tomcat有兩種方式:
- 雙擊bin/startup.bat
- 運行bin/catalina.bat run
它們對應於Bootstrap與Catalina兩個類,咱們如今只關心Catalina這個類,這個類使用Apache Digester解析conf/server.xml文件生成tomcat組件,而後再調用Embedded類的start方法啓動tomcat。
因此,集成Tomcat的方式就有如下兩種了:
- 沿用tomcat自身的server.xml
- 本身定義一個xml格式來配置tocmat的各參數,本身再寫解析這段xml,而後使用tomcat提供的API根據這些xml來生成Tomcat組件,最後調用Embedded類的start方法啓動tomcat
我的以爲第一種方式要優越,給開發者比較好的用戶體驗,若是使用這種,直接模仿Catalina類的方法便可實現集成。
目前,JOnAS就使用了這種集成方式,JBoss、GlassFish使用的第二種自定義XML的方式。
3.2 Connector類圖與順序圖
圖3-1 Connector相關類圖
圖3-2 Connector工做流程順序圖
從上面二圖中咱們能夠獲得以下信息:
- Tomcat中有四種容器(Context、Engine、Host、Wrapper),前三者常見,第四個不常見但它也是實現了Container接口的容器
- 若是要自定義一個Connector的話,只須要實現ProtocolHander接口,該接口定義以下:
圖3-3 自定義connector時需實現的ProtocolHandler接口
Tomcat以HTTP(包括BIO與NIO)、AJP、APR、內存四種協議實現了該接口(它們分別是:AjpAprProtocol、AjpProtocol、Http11AprProtocol、Http11NioProtocol、Http11Protocal、JkCoyoteHandler、MemoryProtocolHandler),要使用哪一種Connector就在conf/server.xml中配置,在Connector的構造函數中會經過反射實例化所配置的實現類:
<Connector port="8181"
protocol="org.apache.coyote.http11.Http11AprProtocol " />
3.3 Connector的工做流程
下面咱們以Http11AprProtocol爲例說明Connector的工做流程。
- 它將工做委託給NioEndpoint類。在NioEndpoint類的init方法中構建一個SocketServer(固然,不一樣的實現類會有一些微小的變化,例如若是是NIO,它構建的就是SocketServerChannel)
- 在NioEndpoint.Acceptor類中會接收一個客戶端新的鏈接請求,以下圖:
- 在NioEndpoint類中,有一個內部接口Handle,該接口定義以下:
- 在Http11NioProtocol類中實現了Handle這個內部接口,並調用Http11NioProcessor類(該類實現了ActionHook回調接口)。在Response類中會調用ActionHook實現類的相關方法的,Response類的action方法以下:
- Http11NioProcessor的process實現方法中,會經過Adapter來調用Servler容器生成響應結果。