eclipse添加tomcat_死磕Tomcat系列(1)——整体架构

点击上方“Java技术前线”,选择“置顶或者星标”

与你一起成长

在许多的高端开发的岗位中都会或多或少有要求面试人员要研究过一些常用中间件源码。这是因为一切的秘密都是藏在源码中,阅读源码能够让我们对框架或者中间件的理解更加深刻,而我们也能够在源码的研究中获得其中一些优秀的设计方式。而我们的中间件和源码那么多,我们该从何入手呢?其实大部分的中间件或者框架都有一些共性的部分,例如网络编程、多线程、反射和类加载等技术。所以深入研究透了一两个中间价的话,那么再回过头来看其他的中间件,那么就会很容易理解它里面所用的技术以及原理。而作为一个老牌的WEB端框架Tomcat,无论是其整体的架构设计,还是其内在的一些技术灵活应用,都值得我们一看。

在学习框架的时候,我一般都是对这个框架有一个整体的认识。知道它整体是如何运行的,然后再深入其中某部分进行研究,这样会事半功倍。

整体架构

我们想要了解一个框架,首先要了解它是干什么的,Tomcat我们都知道,是用于处理连接过来的Socket请求的。那么Tomcat就会有两个功能:

  • 对外处理连接,将收到的字节流转化为自己想要的Request和Response对象

  • 对内处理Servlet,将对应的Request请求分发到相应的Servlet中

那么我们整体的骨架就出来了,Tomcat其实就分为两大部分,一部分是连接器(Connnector)处理对外连接和容器(Container)管理对内的Servelet。大体的关系图如下

c82d8b6026a696ee5103d8e84fd6e1a6.png

最外层的大框就是代表一个Tomcat服务,一个Tomcat服务可以对应多个Service。每个Service都有连接器和容器。这些对应的关系我们也可以打开在Tomcat目录配置文件中 server.xml中看出来。

<Server port="8006" shutdown="SHUTDOWN">

<Service name="Catalina">

<Connector port="8080" protocol="HTTP/1.1"

connectionTimeout="20000"

redirectPort="8443" />

<Connector port="8010" protocol="AJP/1.3" redirectPort="8443" />

<Engine name="Catalina" defaultHost="localhost">

<Realm className="org.apache.catalina.realm.LockOutRealm">

<Realm className="org.apache.catalina.realm.UserDatabaseRealm"

resourceName="UserDatabase"/>

Realm>

<Host name="localhost" appBase="webapps"

Host>

Engine>

Service>

Server>

复制代码

这里我将其中配置文件中删除了一些内容精简了一下,这里我们可以看到连接器其实就是 Connector,一个Service中可以有多个连接器,容器其实对应的就是 Engine

Tomcat的整体架构简单来说就是这样的对应关系。接下来我们简单的介绍连接器的整体架构和容器的整体架构。

连接器

我们可以看到上图中连接器传给容器的是 ServletRequest对象,而容器传给连接器的是 ServletResponse对象,这些在网络传输过程中是肯定不行的,因为网络传输中传送的字节流。所以连接器的功能需求我们大概能总结出来以下几点。

  • Socket连接

  • 读取请求网络中的字节流

  • 根据相应的协议(Http/AJP)解析字节流,生成统一的 TomcatRequestt对象

  • 将 TomcatReques传给容器

  • 容器返回 TomcatResponse对象

  • 将 TomcatResponse对象转换为字节流

  • 将字节流返回给客户端

其实上面的细分都能总结为以下的三点

  • 网络通信

  • 应用层协议的解析

  • Tomcat的 Request/Response与 ServletRequest/ServletResponse对象的转化

而在Tomcat中它也用了三个类来实现上面的三个功能,分别对应如下

  • EndPoint

  • Processor

  • Adapter

用图表示他们的关系的话就是这样

ad7df0c0374b5f49b34159b85e39c2e4.png

容器

容器,顾名思义就是装东西的器具,那么这个Tomcat容器是装什么的呢?其实主要的就是装了Servlet的。那么容器是如何设计的呢?Tomcat的容器设计其实是用了组合设计模式(不了解组合设计模式的可以看我之前的文章不学无数——组合模式)。其实从 Server.xml中我们也能看到其关系了。

<Engine name="Catalina" defaultHost="localhost">

<Host name="localhost" appBase="webapps"

unpackWARs="true" autoDeploy="true">

Host>

Engine>

复制代码

在这里面我们只能看到容器中的两个模块,一个是顶层模块 Engine,另一个是 Host,其实还有两个模块,一个是 Context对应的是我们webapp里面的每个应用文件夹,每个文件夹就是对应一个 Context,还有一个模块 Wrapper对应的是我们 Context中的所有servlet, Wrapper管理了访问关系与具体的Servlet的对应。图表示就是下面这样。

5a1f2af71dbfcdc9395e6c6ca68e1ccc.png

Tomcat中容器所有模块都实现了 Container接口,而组合模式的意义就是使得用户对于单个对象和组合对象的使用具有一致性,即无论添加多少个 Context其使用就是为了找到其下面的Servlet,而无论添加多少个Host也是为了找个下面的Servlet。而在容器中设计了这么多的模块,一个请求过来Tomcat如何找到对应的Servlet进行处理呢?

请求如何定位

我们就举个最简单的例子,我们本机应用上启动了一个Tomcat,webapp下有我们部署的一个应用 buxuewushu。我们在浏览器上输入 http://localhost:8080/buxuewushu/add.do是如何找到对应Servlet进行处理呢?

在我们启动Tomcat的时候,连接器就会进行初始化监听所配置的端口号,这里我们配置的是8080端口对应的协议是HTTP。

  • 请求发送到本机的8080端口,被在那里监听的HTTP/1.1的连接器Connector获得

  • 连接器Connector将字节流转换为容器所需要的 ServletRequest对象给同级 Service下的容器模块Engine进行处理

  • Engine获得地址 http://localhost:8080/buxuewushu/add。匹配他下面的Host主机

  • 匹配到名为localhost的Host(就算此时请求为具体的ip,没有配置相应的Host,也会交给名为localhost的Host进行处理,因为他是默认的主机)

  • Host匹配到路径为 /buxuewushu的Context,即在webapp下面找到相应的文件夹

  • Context匹配到URL规则为*.do的servlet,对应为某个Servlet类

  • 调用其 doGet或者 doPost方法

  • Servlet执行完以后将对象返回给Context

  • Context返回给Host

  • Host返回给Engine

  • Engine返回给连接器Connector

  • 连接器Connector将对象解析为字节流发送给客户端

f3f7f19241621ac5416752c0dba78dcc.png

往期关于Tomcat的文章

  • 一次奇怪的StackOverflowError问题查找之旅

  • 如何断点调试Tomcat源码

参考

  • 深入拆解Tomcat&Jetty

作者:不学无数的程序员

来源:https://juejin.im/post/5d007570f265da1b8b2b5671

热门内容:

  • 888G面试资源分享
  • Mybatis教程1:MyBatis快速入门
  • MyBatis教程2:使用MyBatis对表执行CRUD操作
  • MyBatis教程3:优化MyBatis配置文件中的配置
  • MyBatis教程4:解决字段名与实体类属性名不相同的冲突
  • 2019年Java经典面试题汇总
  • Maven教程1:Maven入门
  • Maven教程2:Maven项目构建过程练习
  • Maven教程3:使用Maven构建项目
  • Maven教程4:Maven核心概念
  • Maven教程5: 聚合与继承
  • Maven教程6: Maven与Eclipse整合
  • Maven教程7:eclipse中使用Maven创建Web项目.md
  • Maven教程8: 使用Maven构建多模块项目
  • Maven教程9: 使用Nexus搭建Maven私服

846c95ef0226cb4656b7fcf46e4b5093.png

喜欢就点个"在看"呗^_^

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值