黑马程序员技术交流社区
标题:
【武汉校区】Tomcat 工作流程(二)
[打印本页]
作者:
武汉分校-小舞
时间:
2019-3-28 16:16
标题:
【武汉校区】Tomcat 工作流程(二)
Connector 是什么?
在
tomcat
中,connector 是用于负责接收来自客户端的连接,其主要任务是负责处理浏览器发送过来的请求,并交由后续的代码进行处理。connector 对象持有 ProtocolHandler 对象;ProtocolHandler 对象持有 AbstractEndpoint 对象。
AbstractEndpoint 负责创建服务器套接字,并绑定到监听端口;同时还创建accepter 线程来接收客户端的连接以及 poller 线程来处理连接中的读写请求。
1,实例化 Connector,构造一个 Connector 对象 2,调用 Connector 的 initIntenal方法,初始化 Connetor 3,调用 ProtocolHanlder 的 init 方法,完成
ProtocolHanlder 的初始化。这个过程包括了创建线程池并创建一个线程处理浏览器请求 4,调用 Connector 的 startIntenal 方法,启动 Connector 5,调用
ProtocolHandler 的 start 方法,启动 Protocolhanlder
Connector 的配置
既然是处理浏览器请求,那么需要支持 http 协议,在 Tomcat 中有两种协议处理器:HTTP/1.1 与 AJP/1.3 协议处理器。在 server.xml 中已经指明 tomcat 所支
持的两种协议:
<Connector
port
=
"8080"
protocol
=
"HTTP/1.1"
connectionTimeout
=
"20000"
redirectPort
=
"8443"
/>
<Connector
port
=
"8009"
protocol
=
"AJP/1.3"
redirectPort
=
"8443"
/>
Connector 的内部初始化
Connector 的创建过程主要是初始化 ProtocolHandler。server.xrnl 配置文件中Connector 标签的 protocol 属性会设置到 Connector 构造函数的参数中,它用于指定 ProtocolHandler 的类型, Connector 的构造函数代码如下:
public
Connector(
String
protocol) {
setProtocol(protocol);
// Instantiate protocol handler
try
{
Class
<?>
clazz
=
Class.forName(protocolHandlerClassName);
this.
protocolHandler = (ProtocolHandler) clazz.newInstance();
}
catch
(Exception e) {
log.error(sm.getString(
"coyoteConnector.protocolHandlerInstantiationFailed"
), e);
}
}
这里首先根据传入的 protocol 参数调用 setProtocol 方法设置了
protocolHandlet℃lassName 属性,接着用 protoco!HandlerClassName 所代表
的类创建了ProtocolHandler 并赋值给了protocolHandler属性。setProtocol 方
法代码如下:
public
void
setProtocol
(
String
protocol) {
if
(AprLifecycleListener.isAprAvailable()) {
if
(
"HTTP/1.1".
equals(protocol)) {
setProtocolHandlerClassName
(
"org.apache.coyote.http11.Http11AprProtocol"
);
}
else if
(
"AJP/1.3"
.equals(protocol)) {
setProtocolHandlerClassName
(
"org.apache.coyote.ajp.AjpAprProtocol"
);
}
else if
(protocol
!=
null
) {
setProtocolHandlerClassName(protocol);
}
else
{
setProtocolHandlerClassName
(
"org.apache.coyote.http11.Http11AprProtocol"
);
}
}
else
{
if
(
"HTTP/1.1"
.equals(protocol)) {
// tomcat8 默认配置
setProtocolHandlerClassName
(
"org.apache.coyote.http11.Http11NioProtocol"
);
}
else if
(
"AJP/1.3"
.equals(protocol)) {
setProtocolHandlerClassName
(
"org.apache.coyote.ajp.AjpNioProtocol"
);
}
else if
(protocol
!=
null
) {
setProtocolHandlerClassName(protocol);
}
}
}
在创建 Http11NioProtocol 实例的时候,会创建 NioEndpoint、
Http11ConnectionHandler 实例。
public
Http11NioProtocol() {
// 创建 nioEndPoint
endpoint
=new
NioEndpoint();
// 创建 Http11ConnectionHandler
cHandler
= new
Http11ConnectionHandler(
this
);
((NioEndpoint) endpoint).setHandler(cHandler);
// 是指 socket 被关闭时逗留的时间,值为-1。
// 在这段时间内,socket 会尽量把未送出去的数据给发出去。
setSoLinger(Constants.DEFAULT_CONNECTION_LINGER);
// 设置读取数据超时
setSoTimeout(Constants.DEFAULT_CONNECTION_TIMEOUT);
// 设置 tcp_nodelay,
setTcpNoDelay(Constants.DEFAULT_TCP_NO_DELAY);
}
NioEndpoint 是 Connector 中处理客户端连接的核心类,负责创建服务器套接字,并绑定到监听端口;同时还创建 accepter 线程来接收客户端的连接以及 poller
线程来处理连接中的读写请求。
当 Connector 调用 init()方法时本质是调用 initInternal(),initInternal()方法又会最终调用 endpoint 的 init()方法
NioEndpoint 的 bind 方法
public
void
bind
()
throws
Exception {
// 打开 serverSocketChannel
serverSock = ServerSocketChannel.open();
// 设置 socket 属性
socketProperties.setProperties(serverSock.socket());
InetSocketAddress addr
=
(getAddress()
!=
null
?
new
InetSocketAddress(getAddress(),getPort()):
new
InetSocketAddress(getPort()));
// 绑定监听端口
serverSock.socket().bind(addr,getBacklog());
// 设为阻塞模式
serverSock.configureBlocking(true); //mimic APR behavior
// 设置超时
serverSock.socket().setSoTimeout(getSocketProperties().getSoTimeout());
//.....省略
}
当 Connector 调用 start()方法时会执行 startInternal()方法,startInternal()方法则
调用 endpoint 的 start()方法。 endpoint 的 start()方法是在 AbstractEndPoint 中
实现的,并调用推迟到 NioEndPoint 中的 startInternal()方法。
@Override
public
void
startInternal
()
throws
Exception {
if
(
!
running) {
running
=
true
;
paused
=
false
;
// 创建缓存容器
processorCache
=
new
SynchronizedStack
<>
(SynchronizedStack.DEFAULT_SIZE,
socketProperties.getProcessorCache());
eventCache
=
new
SynchronizedStack
<>
(SynchronizedStack.DEFAULT_SIZE,
socketProperties.getEventCache());
nioChannels
=
new
SynchronizedStack
<>
(SynchronizedStack.DEFAULT_SIZE,
socketProperties.getBufferPool());
// 创建线程池
if
( getExecutor()
==
null
) {
createExecutor();
}
// 初始化计数器 Latch
initializeConnectionLatch();
// 创建 Poller 线程
pollers
=
new
Poller[getPollerThreadCount()];
for
(i
nt
i
=
0
; i
<
pollers.length; i
++
) {
pollers
=
new
Poller();
Thread pollerThread
=
new
Thread(pollers
, getName()
+ "-ClientPoller-"+
i);
pollerThread.setPriority(threadPriority);
pollerThread.setDaemon(
true
);
pollerThread.start();
}
// 创建 Acceptor 线程
startAcceptorThreads();
}
}
最重要的是创建 Poller 和 Acceptor 线程。Acceptor 线程处理
serverSocketChannel 的请求接收事件;Poller 处理 serverSocketChannel 的读
写事件。
protected class Acceptor extends AbstractEndpoint.Acceptor {
@Override
public void run() {
int errorDelay = 0;
//......
try {
//通过同步计数器来限制连接数目
//当连接数目超过上限时,则等待
//其中同步计算器是通过继承 AQS 实现的
//默认的最大连接数是 10000
countUpOrAwaitConnection();
SocketChannel socket = null;
try {
//接收连接,此处并不是使用 selector 实现
//在前面的代码中已知 serverSock 是阻塞模式的。
socket = serverSock.accept();
} catch (IOException ioe) {
//we didn't get a socket
countDownConnection();
// Introduce delay if necessary
errorDelay = handleExceptionWithDelay(errorDelay);
// re-throw
throw ioe;
}
// Successful accept, reset the error delay
errorDelay = 0;
// setSocketOptions() will add channel to the poller
// if successful
if (running && !paused) {
// 在 setSocketOptions 中将接收到的 socket 传给 poller 线程进行处理
if (!setSocketOptions(socket)) {
countDownConnection();
closeSocket(socket);
}
} else {
countDownConnection();
closeSocket(socket);
//.....
}
Acceptor 线程会反复执行 serverSock.accept()等待客户端连接的到来,等接收到一个客户端连接时,会把接收到的 socket 传给后续的 poller 线程处理,其执行过程在 setSocketOptions()方法中。
欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/)
黑马程序员IT技术论坛 X3.2