Tomcat 架构探索
前言
花了一个礼拜的时间阅读了 how tomcat works
,本文基于此书,整理了一下Tomcat 5
的基本架构,其实也没什么多复杂的东西,无非是解析Http
请求,然后调用相应的Servlet
。另推荐看CSAPP
的网络编程那一章
顺便问问有无需要暑假实习
的?坐标杭州
,Java后台
方向。有意可联系我 threezj@hotmail.com
基本架构
Tomcat
由两个模块协同合作
connector
container
connector
负责解析处理HTTP
请求,比如说请求头
,查询字符串
,请求参数
之类的。生成HttpRequest
和HttpResponse
之后交给container
,由它负责调用相应的Servlet
。
Connector
Tomcat
默认的Connector
为HttpConnector
。作为Connector
必须要实现Connector
这个接口。
Tomcat
启动以后会开启一个线程,做一个死循环,通过ServerSocket
来等待请求。一旦得到请求,生成Socket
,注意这里HttpConnector
并不会自己处理Socket
,而是把它交给HttpProcessor
。详细看下面代码,这里我只保留了关键代码。
public void run() { // Loop until we receive a shutdown command while (!stopped) { Socket socket = null; try { socket = serverSocket.accept(); //等待链接 } catch (AccessControlException ace) { log("socket accept security exception", ace); continue; } // Hand this socket off to an appropriate processor HttpProcessor processor = createProcessor(); processor.assign(socket); //这里是立刻返回的 // The processor will recycle itself when it finishes } }
注意一点,上面的processor.assign(socket);
是立刻返回的,并不会阻塞在那里等待。因为Tomcat不可能一次只能处理一个请求,所以是异步的,每个processor
处理都是一个单独的线程。
HttpProcessor
上面的代码并没有显示调用HttpProcessor
的process
方法,那这个方法是怎么调用的呢?我们来看一下HttpProcessor
的run
方法。
public void run() { // Process requests until we receive a shutdown signal while (!stopped) { // Wait for the next socket to be assigned Socket socket = await(); if (socket == null) continue; // Process the request from this socket try { process(socket); } catch (Throwable t) { log("process.invoke", t); } // Finish up this request connector.recycle(this); } }
我们发现他是调用await
方法来阻塞等待获得socket
方法。而之前Connector
是调用assign
分配的,这是什么原因?
下面仔细看await
和assign
方法。这两个方法协同合作,当assign
获取socket
时会通知await
然后返回socket
。
synchronized void assign(Socket socket) { // Wait for the Processor to get the previous Socket while (available) { try { wait(); } catch (InterruptedException e) { } } // Store the newly available Socket and notify our thread this.socket = socket; available = true; notifyAll(); } private synchronized Socket await() { // Wait for the Connector to provide a new Socket while (!available) { try { wait(); } catch (InterruptedException e) { } } // Notify the Connector that we have received this Socket Socket socket = this.socket; available = false; notifyAll(); return (socket); }
默认available
为false
。
接下来就是剩下的事情就是解析请求,填充HttpRequest
和HttpResponse
对象,然后交给container
负责。
这里我不过多赘述如何解析
private void process(Socket socket) { //parse .... connector.getContainer().invoke(request, response); .... }
Container
A Container is an object that can execute requests received from a client, and return responses based on those requests
Container
是一个接口,实现了这个接口的类的实例,可以处理接收的请求,调用对应的Servlet
。
总共有四类Container
,这四个Container
之间并不是平行关系,而是父子关系
Engine
– 最顶层的容器,可以包含多个Host
Host
– 代表一个虚拟主机,可以包含多个Context
Context
– 代表一个web应用
,也就是ServletContext
,可以包含多个Wrappers
Wrapper
– 代表一个Servlet
,不能包含别的容器了,这是最底层
Container的调用
容器好比是一个加工厂,加工接受的request
,加工方式和流水线也很像,但又有点区别。这里会用到一个叫做Pipeline
的 东西,中文翻译为管道
,request
就放在管道里顺序加工,进行加工的工具叫做Valve
,好比手术刀,Pipeline
可添加多个Valve
,最后加工的工具称为BaseValve
上面可能讲的比较抽象,接下来我们来看代码。Engine
是顶层容器,所以上面invoke
,执行的就是Engine
的方法。StandardEngine
是Engine
的默认实现,注意它也同时实现了Pipeline
接口,且包含了Pipeline
。
它的构造方法同时指定了baseValve
,也就是管道最后一个调用的Valve
public StandardEngine() { super(); pipeline.setBasic(new StandardEngineValve()); }
好,接着我们看invoke
,这个方法是继承自ContainerBase
。只有一行,之间交给pipeline
,进行加工。
public void invoke(Request request, Response response) throws IOException, ServletException { pipeline.invoke(request, response); }
下面是StandardPipeline
的invoke
实现,也就是默认的pipeline
实现。
public void invoke(Request request, Response response) throws IOException, ServletException { // Invoke the first Valve in this pipeline for this request (new StandardPipelineValveContext()).invokeNext(request, response); }
也只有一行!调用StandardPipelineValveContext
的invokeNext
方法,这是一个pipeline
的内部类。让我们来看
具体代码
public void invokeNext(Request request, Response response) throws IOException, ServletException { int subscript = stage; stage = stage + 1; // Invoke the requested Valve for the current request thread if (subscript < valves.length) { valves[subscript].invoke(request, response, this); //加工 } else if ((subscript == valves.length) && (basic != null)) { basic.invoke(request, response, this); } else { throw new ServletException (sm.getString("standardPipeline.noValve")); } }
它调用了pipeline
所用的Valve
来对request
做加工,当Valve执行完,会调用BaseValve
,也就是上面的StandardEngineValve
,
我们再来看看它的invoke
方法
// Select the Host to be used for this Request StandardEngine engine = (StandardEngine) getContainer(); Host host = (Host) engine.map(request, true); if (host == null) { ((HttpServletResponse) response.getResponse()).sendError (HttpServletResponse.SC_BAD_REQUEST, sm.getString("standardEngine.noHost", request.getRequest().getServerName())); return; } // Ask this Host to process this request host.invoke(request, response);
它通过(Host) engine.map(request, true);
获取所对应的Host
,然后进入到下一层容器中继续执行。后面的执行顺序
和Engine
相同,我不过多赘述
执行顺序小结
经过一长串的invoke
终于讲完了第一层容器的执行顺序。估计你们看的有点晕,我这里小结一下。
Connector -> HttpProcessor.process() -> StandardEngine.invoke() -> StandardPipeline.invoke() ->
StandardPipelineValveContext.invokeNext() -> valves.invoke() -> StandardEngineValve.invoke() ->
StandardHost.invoke()
到这里位置Engine
这一层结束。接下来进行Host
,步骤完全一致
StandardHost.invoke() -> StandardPipeline.invoke() ->
StandardPipelineValveContext.invokeNext() -> valves.invoke() -> StandardHostValve.invoke() ->
StandardContext.invoke()
然后再进行Context
这一层的处理,到最后选择对应的Wrapping
执行。
Wrapper
Wrapper
相当于一个Servlet
实例,StandardContext
会更根据的request
来选择对应的Wrapper
调用。我们直接来看看
Wrapper
的basevalve
是如果调用Servlet
的service
方法的。下面是StandardWrapperValve
的invoke
方法,我省略了很多,
只看关键。
public void invoke(Request request, Response response, ValveContext valveContext) throws IOException, ServletException { // Allocate a servlet instance to process this request if (!unavailable) { servlet = wrapper.allocate(); } // Create the filter chain for this request ApplicationFilterChain filterChain = createFilterChain(request, servlet); // Call the filter chain for this request // NOTE: This also calls the servlet's service() method String jspFile = wrapper.getJspFile(); //是否是jsp if (jspFile != null) sreq.setAttribute(Globals.JSP_FILE_ATTR, jspFile); else sreq.removeAttribute(Globals.JSP_FILE_ATTR); if ((servlet != null) && (filterChain != null)) { filterChain.doFilter(sreq, sres); } sreq.removeAttribute(Globals.JSP_FILE_ATTR); }
首先调用wrapper.allocate()
,这个方法很关键,它会通过反射
找到对应servlet
的class
文件,构造出实例返回给我们。然后创建一个FilterChain
,熟悉j2ee
的各位应该对这个不陌生把?这就是我们在开发web app
时使用的filter
。然后就执行doFilter
方法了,它又会调用internalDoFilter
,我们来看这个方法
private void internalDoFilter(ServletRequest request, ServletResponse response) throws IOException, ServletException { // Call the next filter if there is one if (this.iterator.hasNext()) { ApplicationFilterConfig filterConfig = (ApplicationFilterConfig) iterator.next(); Filter filter = null; filter = filterConfig.getFilter(); filter.doFilter(request, response, this); return; } // We fell off the end of the chain -- call the servlet instance if ((request instanceof HttpServletRequest) && (response instanceof HttpServletResponse)) { servlet.service((HttpServletRequest) request, (HttpServletResponse) response); } else { servlet.service(request, response); } }
终于,在这个方法里看到了service
方法,现在你知道在使用filter
的时候如果不执行doFilter
,service
就不会执行的原因了把。
小结
Tomcat
的重要过程应该都在这里了,还值得一提的是LifeCycle
接口,这里所有类几乎都实现了LifeCycle
,Tomcat
通过它来统一管理容器的生命流程,大量运用观察者模式。有兴趣的同学可以自己看书
本文文字及图片出自 threezj.com
你也许感兴趣的:
- ThoughtWorks CTO:2025 年之前,我们会看到架构的演进,但不会看到革命
- 苏宁易购移动架构演进的青铜时代、白银时代和黄金时代
- 撇开代码不说,谈谈我对架构的6个冷思考
- WEB前端MVC架构变形记
- 如何为你的开源项目选择正确的品牌架构
- 我理解的架构师
- 在首席架构师眼里,架构的本质是……
- PHP系统设计与云架构
- 具有魔法的 H.264
- 多用户环境中的 rootless Docker
你对本文的反应是: