参考文章
https://time.geekbang.org/column/article/96328
https://time.geekbang.org/column/article/96764
一、目标:
- 学到 Tomcat 的总体架构,学会从宏观上怎么去设计一个复杂系统,怎么设计顶层模块,以及模块之间的关系;
- 为我们深入学习 Tomcat 的工作原理打下基础。
二、Tomcat 总体架构
首先是要了解需求。 Tomcat 要实现 2 个核心功能:
- 处理 Socket 连接,负责网络字节流与 Request 和 Response 对象的转化。
- 加载和管理 Servlet,以及具体处理 Request 请求。
因此 Tomcat 设计了两个核心组件连接器(Connector)和容器(Container)来分别做这两件事情。连接器负责对外交流,容器负责内部处理。
Tomcat 支持的 I/O 模型有:
- NIO:非阻塞 I/O,采用 Java NIO 类库实现。
- NIO.2:异步 I/O,采用 JDK 7 最新的 NIO.2 类库实现。
- APR:采用 Apache 可移植运行库实现,是 C/C++ 编写的本地库。
Tomcat 支持的应用层协议有:
- HTTP/1.1:这是大部分 Web 应用采用的访问协议。
- AJP:用于和 Web 服务器集成(如 Apache)。
- HTTP/2:HTTP 2.0 大幅度的提升了 Web 性能。
Tomcat 为了实现支持多种 I/O 模型和应用层协议,一个容器可能对接多个连接器。但是单独的连接器或者容器都不能对外提供服务,需要把它们组装起来才能工作,组装后这个整体叫作 Service 组件。
这里请你注意,Service 本身没有做什么重要的事情,只是在连接器和容器外面多包了一层,把它们组装在一起。Tomcat 内可能有多个 Service,这样的设计也是出于灵活性的考虑。通过在 Tomcat 中配置多个 Service,可以实现通过不同的端口号来访问同一台机器上部署的不同应用。
从图上你可以看到,最顶层是 Server,这里的 Server 指的就是一个 Tomcat 实例。一个 Server 中有一个或者多个 Service,一个 Service 中有多个连接器和一个容器。连接器与容器之间通过标准的 ServletRequest 和 ServletResponse 通信。
二、连接器
连接器对 Servlet 容器屏蔽了协议及 I/O 模型等的区别,无论是 HTTP 还是 AJP,在容器中获取到的都是一个标准的 ServletRequest 对象
- ProtocolHandle 处理socket连接和应用层协议 Http11NioProtocol 和 AjpNioProtocol
- Endpoint
处理socket连接和发送 用来实现 TCP/IP 协议 EndPoint利用Socket接口来将底层传来的数据转化成为HTTP格式的数据
AbstractEndpoint 的具体子类,比如 NioEndpoint 和 Nio2Endpoint 中有两个子组件Acceptor 用于监听 Socket 连接请求。SocketProcessor 用于处理接收到的socket请求,实现了run方法。 - Processor
处理应用层协议 用来实现 HTTP 协议
AbstractProcessor 实现类AjpProcessor、Http11Processor
Endpoint 接收到 Socket 连接后,生成一个 SocketProcessor 任务提交到线程池去处理,SocketProcessor 的 run 方法会调用 Processor 组件去解析应用层协议,Processor 通过解析生成 Request 对象后,会调用 Adapter 的 Service 方法。
- Adapter
CoyoteAdapter
连接器调用 CoyoteAdapter 的 sevice 方法,传入的是 Tomcat Request 对象,CoyoteAdapter 负责将 Tomcat Request 转成 ServletRequest,再调用容器的 service 方法。 适配器模式
Endpoint 接收到 Socket 连接后,生成一个 SocketProcessor 任务提交到线程池去处理,SocketProcessor 的 run 方法会调用 Processor 组件去解析应用层协议,Processor 通过解析生成 Request 对象后,会调用 Adapter 的 Service 方法
三、容器
- 请求定位过程
- 由协议和端口号确定service
- 由域名选定host
- 由url选定context组件
- 由url选定Wrapper(Servlet)
- engine host wrapper servlet的调用流程是怎么实现的 责任链模式
1.通过Pipeline-Value管道来实现的,他是一种责任链模式。Valve接口和PipleLine接口。
2.org.apache.catalina.Pipeline#getBasic方法表示调用下一层容器的PipleLine的第一个Value.
3.Wrapper 容器的最后一个 Valve 会创建一个 Filter 链,并调用 doFilter 方法,最终会调到 Servlet 的 service 方法。
//链表
public interface Valve {
public Valve getNext();
public void setNext(Valve valve);
//触发下一个的Value调用
public void invoke(Request request, Response response)
}
public interface Pipeline extends Contained {
public void addValve(Valve valve);
//调用下一层容器的PipleLine的第一个Value.
public Valve getBasic();
public void setBasic(Valve valve);
public Valve getFirst();
}