Tomcat8源码笔记(六)连接器Connector分析
根据 Tomcat8源码笔记(五)组件Container分析 前文分析,StandardService的初始化重心由 StandardEngine转移到了Connector的初始化,本篇记录下Connector实例化到初始化的过程
国际惯例这个词最近很常见,那国际惯例,我先放上Tomcat各个组件结构图:Connector地位,一个Service下可以有多个Connector,和Container愉快的生活在Service里;

先来看看server.xml中对于Connector元素的位置以及如何定义:
一个server.xml中确实可以定义多个Connector,按照官方注释的意思,一个连接器Connector代表了一个接受请求返回响应的端点,请求类型可以是HTTP、AJP这种;

废话少说,先来看下Tomcat如何解析Connector元素,并且生成怎么样的结构
如果仔细看了前面几篇博客,直接进入Catalina的createStartDigester方法,找到Connector即可:

如果你不太了解Rule规则如何解析的,Tomcat8源码笔记(三)Catalina加载过程 也许可以帮你了解这是一个怎么样的流程; 从上面定义的规则,可以得到几点:
1.ConnectorCreateRule这肯定是个动态的创建Connector的规则,与以往定义好了对象类型不一样.
2.Connector实例化后会被Service调用addConnector,关联起来;或许还会有双向关联藏在哪里悄悄实现了.
3.Connector也允许添加Listener子元素,监听器实现特殊的需求;
Connector实例化
按照经验,每个Connector实例化都会是这个逻辑:顺序调用rules的Rule对象的begin方法,然后顺序调用rules的Rule对象的body方法,最后逆序调用rules的Rule对象的end方法;

Step1.先从第一个规则ConnectorCreateRule的begin方法入手

Connector很多逻辑都在构造器中调用了,所以这里不得不先详细介绍下Connector构造器的方法:这里就以最常见的HTTP/1.1这种连接器来记录

首先是setProtocol方法:
根据不同协议,初始化连接器的不同protocolClassName,HTTP/1.1在Tomcat8.0中使用的是Http11NioProtocol

其次是通过反射调用Http11NioProtocol的空参构造器实例化一个ProtocolHandler,最终目的是作为Connector的protocolHandler;果不其然,逻辑又在Http11NioProtocol的空参构造器中出现了!

Http11NioProtocol初始化
endPoint初始化摸不清头脑

Http11ConnectionHandler初始化为了持有Http11NioProtocol,用来专门做一些处理; 还让endPoint持有了Http11ConnectionHandler,Http11NioProtocol初始化最后给endPoint设置了一堆属性,其实就是给endPoint的socketProperties属性设置。

最终,Http11NioProtocol初始化就完成了这样的结构:三者相互存在引用关系

Step2.Connector实例化的第二个规则说起,setAllPropertiesRule
setAllPropertiesRule用来完成给Connector的属性赋值工作,比如常见的下面配置,就需要给Connector设置四个属性:port端口,protocol协议,connectionTimeout,redirectPort;
port直接是Connector的属性设置即可,protocol之前也见过setProtocol方法,就是设置了protocolHandlerClassName属性,所以这一步重复了但是一定会执行做;redirectPort也是Connector属性,直接反射赋值; connectionTimeout不是Connector属性,但是最终也有用啊,最后给了endPoint的socketProperties的 soTimeout属性!

Step3. 由于SetNextRule方法begin为空,所以直接Connector最基本的初始化差不多完成了! 而Rule规则,解析完成server.xml的一个元素就会按照end方法来逆序调用,所以这里起作用的是SetNextRule的end方法,而Tomcat8源码笔记(三)Catalina加载过程 这里记录过 SetNextRule的规则,在这里,就是给StandardService 调用addConnector添加连接器,所以StandardService的Connector默认情况就是两个了,HTTP/1.1和AJP/1.3这两种。
Connector初始化
终于进入到了Connector初始化,初始化了决定了Tomcat有资格接受HTTP等请求,Connector默认是没有监听器,所以初始化initInternal调用之前的触发监听统统忽略,直接进入正题 initInternal方法。

Connector的parseBodyMethods为“POST”,parseBodyMethodsSet为{“POST”}.
核心的就是Http11NioProtocol的初始化
初始化了sslImplementation为了后续操作,调用父类的init方法.

Http11NioProtocol调用的父类初始化方法
给NioEndPoint设置名称,http-nio-8080,这在日志信息中经常出现;最后就是调用NioEndPoint的初始化方法

Connector的初始化由Http11NioProtocol初始化,交给了NioEndPoint初始化
NioEndPoint方法初始化没有交给别的组件来完成了! 自己调用自己的bind方法

bind方法
典型的NIO编程,如果不熟悉的记得多看看NIO视频充下电,我也要抽空充下电。因为NioEndPoint端口设置过了8080,(这里留个悬念,我们只设置了Connector的8080端口,没设置NioEndPoint啊),另外给SocketChannel的socket都设置超时为 server.xml中配置的20000毫秒, 这是NIO编程部分得出的结论。 另外 acceptorThreadCount 是Acceptor线程的个数,pollerThreadCount是Poller线程的个数,poller线程个数取决于 Math.min(2,Runtime.getRuntime().availableProcessors()) ,这两个参数比较重要,后面会有Accpetor线程以及Poller线程具体分析。 最后调用NioSelectorPool的open方法.

NioSelectorPool的Open方法:

NioBlockingSelector的open方法:
实例化BlockPoller对象,肯定是Thread的子类,作为守护线程运行,线程名称为:NioBlockingSelector.BlockPoller-1 , 啥时候用到我们再分析。

至此,Connector连接器我们就介绍了,总结下Connector初始化调用protocolHandler#init方法,protocol#init方法调用endPoint#init初始化,而endPoint初始化调用bind方法,采用NIO编程,由此可见Tomcat HTTP1.1在Tomcat8中使用NIO编程来接受请求。 到这里Tomcat初始化工作都完成了,后面就是Tomcat启动并且接受请求的过程了,可见Tomcat部署项目、接受请求等等一系列都是启动中完成的,初始化只是准备工作。
Tomcat8源码笔记(六)连接器Connector分析的更多相关文章
- Tomcat8源码笔记(五)组件Container分析
Tomcat8源码笔记(四)Server和Service初始化 介绍过Tomcat中Service的初始化 最先初始化就是Container,而Container初始化过程是咋样的? 说到Contai ...
- Tomcat8源码笔记(七)组件启动Server Service Engine Host启动
一.Tomcat启动的入口 Tomcat初始化简单流程前面博客介绍了一遍,组件除了StandardHost都有博客,欢迎大家指文中错误.Tomcat启动类是Bootstrap,而启动容器启动入口位于 ...
- Tomcat8源码笔记(三)Catalina加载过程
之前介绍过 Catalina加载过程是Bootstrap的load调用的 Tomcat8源码笔记(二)Bootstrap启动 按照Catalina的load过程,大致如下: 接下来一步步分析加载过程 ...
- Tomcat8源码笔记(四)Server和Service初始化
上一章 简单说明下Tomcat各个组件: Server:服务器,Tomcat服务器,一个Tomcat只有一个Server组件; Service:业务层,是Server下最大的子容器,一个Server可 ...
- Tomcat8源码笔记(八)明白Tomcat怎么部署webapps下项目
以前没想过这么个问题:Tomcat怎么处理webapps下项目,并且我访问浏览器ip: port/项目名/请求路径,以SSM为例,Tomcat怎么就能将请求找到项目呢,项目还是个文件夹类型的? Tom ...
- Tomcat8源码笔记(一)Lifecycle接口
第一次阅读Tomcat8源码,就以Lifecycle作为笔记阅读的开篇吧,一千个读者就有一千个哈姆雷特,每个人都Tomcat的理解都不同,如果不记录一次Tomcat源码可能忘了就忘了. 断断DEBUG ...
- Tomcat8源码笔记(二)Bootstrap启动
TOMCAT源码调试入口是Bootstrap类的main方法,我的启动参数VM: -Dcatalina.home=E:/Tomcat_Source_Code/apache-tomcat-8.0.53- ...
- Tomcat8源码笔记(九)组件StandardContext启动流程--未完待续
StandardContext代表的是webapps下项目,一个项目就是一个StandardContext,作为Tomcat组件的一部分,就会实现Lifecycle接口,被Tomcat管理着生命周期, ...
- 厉害!这份阿里面试官 甩出的Spring源码笔记,GitHub上已经爆火
前言 时至今日,Spring 在 Java 生态系统与就业市场上,面试出镜率之高,投产规模之广,无出其右.随着技术的发展,Spring 从往日的 IoC 框架,已发展成 Cloud Native 基础 ...
随机推荐
- 二叉树/DFS总结
二叉搜索树(Binary Search Tree,又名排序二叉树,二叉查找树,通常简写为BST)定义如下: 空树或是具有下列性质的二叉树: ()若左子树不空,则左子树上所有节点值均小于或等于它的根节点 ...
- charming_memory
Memory Master 一 .Forget遗忘 遗忘似乎是记忆的天敌,但是善用遗忘规律却能帮助我们更好的记忆. 复习的最佳时间是实际材料的1~24小时,最晚不超过2天,复习时间太长,就有一种生疏的 ...
- C++输出格式
C++输出格式 C++中默认输出有效位数是6位,即 则输出: 221.111.11011199967 //6位有效数字,自动截取保存六位1.99967e+006 //六位以上且无法省略显示将会变为指数 ...
- git的命令行操作
1.初始化本地的git仓库git init,代码存放在这里,git会自动对我们的代码进行管理备份. 2.设置用户信息,设置用户名:git config --global user.name " ...
- Hive(一)
1. HIVE概念: Hive:由Facebook开源用于解决海量结构化日志的数据统计. Hive是基于Hadoop的一个数据仓库工具,可以将结构化的数据文件映射为一张表,并提供类SQL查询功能. 本 ...
- weblogic 控制台访问速度很慢的解决方案
实际是JVM在Linux下的bug 他想调用一个随机函数 但取不到 暂时的解决办法是 1)较好的解决办法: 在Weblogic启动参数里添加 “- Djava.security.egd=file:/d ...
- 剑指offer面试题25:二叉树中和为某一值的路径
题目:输入一棵二叉树和一个整数,打印出二叉树中结点值的和为输入整数的所有路径.从根节点开始往下一直到叶节点所经过的节点形成一条路径. 解题思路:当使用前序遍历的方式访问某一节点时,把该节点添加到路径上 ...
- MySQL优化技巧
目录 MySQL的特点 数据类型优化 整型类型 小数类型 字符串类型 时间类型 主键类型的选择 特殊类型的数据 索引优化 一个使用Hash值创建索引的技巧 前缀索引 多列索引 聚簇索引 覆盖索引 重复 ...
- 51nod OJ P1008 N的阶乘 mod P
P1008 N的阶乘 mod P OJ:51Nod 链接:"http://www.51nod.com/Challenge/Problem.html#!#problemId=1008" ...
- 每天学点SpringCloud(二):服务注册与发现Eureka
相信看过 每天学点SpringCloud(一):简单服务提供者消费者调用的同学都发现了,在最后消费者调用提供者的时候把提供者的地址硬编码在了代码中,这样的方式肯定是不行的,今天,我们就是要Eureka ...