咱们想要了解一个框架,首先要了解它是干什么的,Tomcat咱们都知道,是用于处理链接过来的Socket请求的。那么Tomcat就会有两个功能:web
那么咱们总体的骨架就出来了,Tomcat其实就分为两大部分,一部分是链接器(Connnector)处理对外链接和容器(Container)管理对内的Servelet。apache
大致的关系图以下:设计模式
描述:浏览器
最外层的大框就是表明一个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
。app
Tomcat的总体架构简单来讲就是这样的对应关系。接下来咱们简单的介绍链接器的总体架构和容器的总体架构。框架
咱们能够看到上图中链接器传给容器的是 ServletRequest
对象,而容器传给链接器的是 ServletResponse
对象,这些在网络传输过程当中是确定不行的,由于网络传输中传送的字节流。webapp
因此链接器的功能需求咱们大概能总结出来如下几点。ide
TomcatRequest
t对象TomcatReques
传给容器TomcatResponse
对象TomcatResponse
对象转换为字节流其实上面的细分都能总结为如下的三点
Request/Response
与 ServletRequest/ServletResponse
对象的转化而在Tomcat中它也用了三个类来实现上面的三个功能,分别对应以下
用图表示他们的关系的话就是这样
容器,顾名思义就是装东西的器具,那么这个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的对应。图表示就是下面这样。
Tomcat中容器全部模块都实现了 Container
接口,而组合模式的意义就是使得用户对于单个对象和组合对象的使用具备一致性,
即不管添加多少个 Context
其使用就是为了找到其下面的Servlet,而不管添加多少个Host也是为了找个下面的Servlet。
而在容器中设计了这么多的模块,一个请求过来Tomcat如何找到对应的Servlet进行处理呢?
咱们就举个最简单的例子,咱们本机应用上启动了一个Tomcat,webapp下有咱们部署的一个应用 buxuewushu
。
咱们在浏览器上输入 http://localhost:8080/buxuewushu/add.do
是如何找到对应Servlet进行处理呢?
在咱们启动Tomcat的时候,链接器就会进行初始化监听所配置的端口号,这里咱们配置的是8080端口对应的协议是HTTP。
ServletRequest
对象给同级 Service
下的容器模块Engine进行处理http://localhost:8080/buxuewushu/add
。匹配他下面的Host主机/buxuewushu
的Context,即在webapp下面找到相应的文件夹doGet
或者 doPost
方法