這篇文章將為大家詳細(xì)講解有關(guān)Tomcat中的容器是怎么處理請求的,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。
成都創(chuàng)新互聯(lián)公司-專業(yè)網(wǎng)站定制、快速模板網(wǎng)站建設(shè)、高性價比臨汾網(wǎng)站開發(fā)、企業(yè)建站全套包干低至880元,成熟完善的模板庫,直接使用。一站式臨汾網(wǎng)站制作公司更省心,省錢,快速模板網(wǎng)站建設(shè)找我們,業(yè)務(wù)覆蓋臨汾地區(qū)。費用合理售后完善,10多年實體公司更值得信賴。
我們繼續(xù)跟著上篇文章Adapter
的源碼,繼續(xù)分析,上篇文章結(jié)尾的源碼如下:
//源碼1.類: CoyoteAdapter implements Adapter public void service(org.apache.coyote.Request req, org.apache.coyote.Response res) throws Exception { Request request = (Request) req.getNote(ADAPTER_NOTES); Response response = (Response) res.getNote(ADAPTER_NOTES); postParseSuccess = postParseRequest(req, request, res, response); if (postParseSuccess) { //check valves if we support async request.setAsyncSupported( connector.getService().getContainer().getPipeline().isAsyncSupported()); // Calling the container connector.getService().getContainer().getPipeline().getFirst().invoke( request, response); } }
上面的源碼的主要作用就是獲取到容器,然后調(diào)用getPipeline()
獲取Pipeline
,最后去invoke
調(diào)用,我們來看看這個Pipeline
是做什么的。
//源碼2.Pipeline接口 public interface Pipeline extends Contained { public Valve getBasic(); public void setBasic(Valve valve); public void addValve(Valve valve); public Valve[] getValves(); public void removeValve(Valve valve); public Valve getFirst(); public boolean isAsyncSupported(); public void findNonAsyncValves(Setresult); } //源碼3. Valve接口 public interface Valve { public Valve getNext(); public void setNext(Valve valve); public void backgroundProcess(); public void invoke(Request request, Response response) throws IOException, ServletException; public boolean isAsyncSupported();
我們從字面上可以理解Pipeline
就是管道,而Valve
就是閥門,實際上在Tomcat中的作用也是和字面意思差不多。每個容器都有一個管道,而管道中又有多個閥門。我們通過后面的分析來證明這一點。
我們看到上面的源碼是Pipeline
和Valve
的接口,Pipeline
主要是設(shè)置Valve
,而Valve
是一個鏈表,然后可以進行invoke
方法的調(diào)用。我們回顧下這段源碼:
//源碼4 connector.getService().getContainer().getPipeline().getFirst().invoke( request, response);
這里是直接獲取容器的管道,然后獲取第一個Valve
進行調(diào)用。我們在之前提到過Valve
是一個鏈表,這里只調(diào)用第一個,也就是可以通過Next去調(diào)用到最后一個。我們再回顧下我們第一篇文章《Tomcat在SpringBoot中是如何啟動的》中提到過,容器是分為4個子容器,分別為Engine
、Host
、Context
、Wrapper
,他們同時也是父級和子級的關(guān)系,Engine
>Host
>Context
>Wrapper
。
我之前提到過,每個容器都一個Pipeline
,那么這個是怎么體現(xiàn)出來的呢?我們看容器的接口源碼就可以發(fā)現(xiàn),Pipeline
是容器接口定義的一個基本屬性:
//源碼5. public interface Container extends Lifecycle { //省略其他代碼 /** * Return the Pipeline object that manages the Valves associated with * this Container. * * @return The Pipeline */ public Pipeline getPipeline(); }
我們知道了每個容器都有一個管道(Pipeline
),管道中有許多閥門(Valve
),Valve
可以進行鏈?zhǔn)秸{(diào)用,那么問題來了,父容器管道中的Valve
怎么調(diào)用到子容器中的Valve
呢?在Pipeline
的實現(xiàn)類StandardPipeline
中,我們發(fā)現(xiàn)了如下源碼:
/** // 源碼6. * The basic Valve (if any) associated with this Pipeline. */ protected Valve basic = null; /** * The first valve associated with this Pipeline. */ protected Valve first = null; public void addValve(Valve valve) { //省略部分代碼 // Add this Valve to the set associated with this Pipeline if (first == null) { first = valve; valve.setNext(basic); } else { Valve current = first; while (current != null) { //這里循環(huán)設(shè)置Valve,保證最后一個是basic if (current.getNext() == basic) { current.setNext(valve); valve.setNext(basic); break; } current = current.getNext(); } } container.fireContainerEvent(Container.ADD_VALVE_EVENT, valve); }
根據(jù)如上代碼,我們知道了basic
是一個管道(Pipeline
)中的最后一個閥門,按道理只要最后一個閥門是下一個容器的第一個閥門就可以完成全部的鏈?zhǔn)秸{(diào)用了。我們用一個請求debug下看看是不是和我們的猜測一樣,我們在CoyoteAdapter
中的service
方法中打個斷點,效果如下:
這里我們可以知道,在適配器調(diào)用容器的時候,也就是調(diào)用Engine
的管道,只有一個閥門,也就是basic,值為StandardEngineValve
。我們發(fā)現(xiàn)這個閥門的invoke方法如下:
//源碼7. public final void invoke(Request request, Response response) throws IOException, ServletException { // Select the Host to be used for this Request Host host = request.getHost(); if (host == null) { // HTTP 0.9 or HTTP 1.0 request without a host when no default host // is defined. This is handled by the CoyoteAdapter. return; } if (request.isAsyncSupported()) { request.setAsyncSupported(host.getPipeline().isAsyncSupported()); } // Ask this Host to process this request host.getPipeline().getFirst().invoke(request, response); }
我們繼續(xù)debug查看結(jié)果如下:
所以這里的basic
實際上將會調(diào)用到Host
容器的管道(Pipeline
)和閥門(Valve
),也就是說,每個容器管道中的basic
是負(fù)責(zé)調(diào)用下一個子容器的閥門。我用一張圖來表示:
這張圖清晰的描述了,Tomcat內(nèi)部的容器是如何流轉(zhuǎn)請求的,從連接器(Connector
)過來的請求會進入Engine
容器,Engine
通過管道(Pieline
)中的閥門(Valve
)來進行鏈?zhǔn)秸{(diào)用,最后的basic
閥門是負(fù)責(zé)調(diào)用下一個容器的第一個閥門的,一直調(diào)用到Wrapper
,然后Wrapper
再執(zhí)行Servlet
。
我們看看Wrapper
源碼,是否真的如我們所說:
//源碼8. public final void invoke(Request request, Response response) throws IOException, ServletException { //省略部分源碼 Servlet servlet = null; if (!unavailable) { servlet = wrapper.allocate(); } // Create the filter chain for this request ApplicationFilterChain filterChain = ApplicationFilterFactory.createFilterChain(request, wrapper, servlet); filterChain.doFilter(request.getRequest(), response.getResponse()); }
看到這里,你可能會說這里明明只是創(chuàng)建了過濾器(Filter
)并且去調(diào)用而已,并沒有去調(diào)用Servlet
,沒錯,這里確實沒有去調(diào)用Servlet
,但是我們知道,過濾器(Filter
)是在Servlet
之前執(zhí)行的,也就是說,filterChain.doFilter
執(zhí)行完之后變會執(zhí)行Servlet
。我們看看ApplicationFilterChain
的源碼是否如我們所說:
//源碼9. public void doFilter(ServletRequest request, ServletResponse response) throws IOException, ServletException { //省略部分代碼 internalDoFilter(request,response); } //源碼10. private void internalDoFilter(ServletRequest request, ServletResponse response) throws IOException, ServletException { //省略部分代碼 // Call the next filter if there is one if (pos < n) { //省略部分代碼 ApplicationFilterConfig filterConfig = filters[pos++]; Filter filter = filterConfig.getFilter(); filter.doFilter(request, response, this); return; } //調(diào)用servlet // We fell off the end of the chain -- call the servlet instance servlet.service(request, response);
通過源碼我們發(fā)現(xiàn),在調(diào)用完所有的過濾器(Filter
)之后,servlet
就開始調(diào)用service
。我們看看servlet
的實現(xiàn)類
這里我們熟悉的HttpServlet
和GenericServlet
是Tomcat
包的類,實際上只有HttpServlet
,因為GenericServlet
是HttpServlet
的父類。后面就是移交給了框架去處理了,Tomcat內(nèi)部的請求已經(jīng)到此是完成了。
我們知道,Tomcat是支持部署多個應(yīng)用的,那么Tomcat是如何支持多應(yīng)用的部署呢?是怎么保證多個應(yīng)用之間不會混淆的呢?要想弄懂這個問題,我們還是要回到適配器去說起,回到service
方法
//源碼11.類:CoyoteAdapter public void service(org.apache.coyote.Request req, org.apache.coyote.Response res) throws Exception { //省略部分代碼 // Parse and set Catalina and configuration specific // request parameters //處理URL映射 postParseSuccess = postParseRequest(req, request, res, response); if (postParseSuccess) { //check valves if we support async request.setAsyncSupported( connector.getService().getContainer().getPipeline().isAsyncSupported()); // Calling the container connector.getService().getContainer().getPipeline().getFirst().invoke( request, response); } }
我們在之前的源碼中只談到了connector.getService().getContainer().getPipeline().getFirst().invoke( request, response)
這段代碼,這部分代碼是調(diào)用容器,但是在調(diào)用容器之前有個postParseRequest
方法是用來處理映射請求的,我們跟進看看源碼:
//源碼12.類:CoyoteAdapter protected boolean postParseRequest(org.apache.coyote.Request req, Request request, org.apache.coyote.Response res, Response response) throws IOException, ServletException { 省略部分代碼 boolean mapRequired = true; while (mapRequired) { // This will map the the latest version by default connector.getService().getMapper().map(serverName, decodedURI, version, request.getMappingData()); //沒有找到上下文就報404錯誤 if (request.getContext() == null) { // Don't overwrite an existing error if (!response.isError()) { response.sendError(404, "Not found"); } // Allow processing to continue. // If present, the error reporting valve will provide a response // body. return true; } }
這里就是循環(huán)去處理Url映射,如果Context
沒有找到,就返回404錯誤,我們繼續(xù)看源碼:
//源碼13.類:Mapper public void map(MessageBytes host, MessageBytes uri, String version, MappingData mappingData) throws IOException { if (host.isNull()) { String defaultHostName = this.defaultHostName; if (defaultHostName == null) { return; } host.getCharChunk().append(defaultHostName); } host.toChars(); uri.toChars(); internalMap(host.getCharChunk(), uri.getCharChunk(), version, mappingData); } //源碼14.類:Mapper private final void internalMap(CharChunk host, CharChunk uri, String version, MappingData mappingData) throws IOException { //省略部分代碼 // Virtual host mapping 處理Host映射 MappedHost[] hosts = this.hosts; MappedHost mappedHost = exactFindIgnoreCase(hosts, host); //省略部分代碼 if (mappedHost == null) { mappedHost = defaultHost; if (mappedHost == null) { return; } } mappingData.host = mappedHost.object; // Context mapping 處理上下文映射 ContextList contextList = mappedHost.contextList; MappedContext[] contexts = contextList.contexts; //省略部分代碼 if (context == null) { return; } mappingData.context = contextVersion.object; mappingData.contextSlashCount = contextVersion.slashCount; // Wrapper mapping 處理Servlet映射 if (!contextVersion.isPaused()) { internalMapWrapper(contextVersion, uri, mappingData); } }
由于上面的源碼比較多,我省略了很多代碼,保留了能理解主要邏輯的代碼,總的來說就是處理Url包括三部分,映射Host
,映射Context
和映射Servlet
(為了節(jié)省篇幅,具體細(xì)節(jié)源碼請感興趣的同學(xué)自行研究)。
這里我們可以發(fā)現(xiàn)一個細(xì)節(jié),就是三個處理邏輯都是緊密關(guān)聯(lián)的,只有Host
不為空才會處理Context
,對于Servlet
也是同理。所以這里我們只要Host
配置不同,那么后面所有的子容器都是不同的,也就完成了應(yīng)用隔離的效果。但是對于SpringBoot內(nèi)嵌Tomcat方式(使用jar包啟動)來說,并不具備實現(xiàn)多應(yīng)用的模式,本身一個應(yīng)用就是一個Tomcat。
為了便于理解,我也畫了一張多應(yīng)用隔離的圖,這里我們假設(shè)有兩個域名admin.luozhou.com
和web.luozhou.com
然后我每個域名下部署2個應(yīng)用,分別是User
,log
,blog
,shop
。那么當(dāng)我去想去添加用戶的時候,我就會請求admin.luozhou.com
域名下的User
的Context
下面的add
的Servlet(說明:這里例子設(shè)計不符合實際開發(fā)原則,add這種粒度應(yīng)該是框架中的controller完成,而不是Servlet)。
這篇文章我們研究了Tomcat中容器是如何處理請求的,我們來回顧下內(nèi)容:
連接器把請求丟給適配器適配后調(diào)用容器(Engine
)
容器內(nèi)部是通過管道(Pieline
)-閥門(Valve
)模式完成容器的調(diào)用的,父容器調(diào)用子容器主要通過一個basic
的閥門來完成的。
最后一個子容器wrapper
完成調(diào)用后就會構(gòu)建過濾器來進行過濾器調(diào)用,調(diào)用完成后就到了Tomcat內(nèi)部的最后一步,調(diào)用servlet。也可以理解我們常用的HttpServlet
,所有基于Servlet
規(guī)范的框架在這里就進入了框架流程(包括SpringBoot)。
最后我們還分析了Tomcat是如何實現(xiàn)多應(yīng)用隔離的,通過多應(yīng)用的隔離分析,我們也明白了為什么Tomcat要設(shè)計如此多的子容器,多子容器可以根據(jù)需要完成不同粒度的隔離級別來實現(xiàn)不同的場景需求。
關(guān)于“Tomcat中的容器是怎么處理請求的”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,使各位可以學(xué)到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。