文章目录


Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_start


启动流程分析

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_start_02

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_初始化_03


Pre

​Tomcat - Tomcat 8.5.55 启动过程源码分析阶段二_load加载初始化​

说完了load阶段,这里我们继续来看下最后一个start阶段


Star阶段

start总览

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_生命周期_04

你会发现和 load阶段非常相似

这里我们就不展开的这么详细了,梳理核心脉络~


start源码分析

Bootstrap#main -----> daemon.start(); -------反射调用-------> Catalina # start -------------> getServer().start(); -------模板方法LifeCycleBase--------> startInternal();

还是模板模式

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_start_05

StandardServer Start

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_tomcat启动分析_06

这里主要是继续启动Service

StandardService Start

还是会走到生命周期里 ,统一收到LifeCycle接口定义的生命周期管理 抽象类LifeCycleBase ,子类重写startInternal()

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_tomcat源码_07

一样的套路

精简后的核心代码如下:

@Override
protected void startInternal() throws LifecycleException {


// Start our defined Container first
if (engine != null) {
synchronized (engine) {
engine.start();
}
}


// Start our defined Connectors second
synchronized (connectorsLock) {
for (Connector connector: connectors) {
connector.start();
}
}
}

分两大块

  • engine.start()
  • connector.start()

接下来逐一分析


StandardEngine Start

大致流程一览

StandardEngine # startInternal ---------> 调用父类ContainerBase#startInternal ---------> startStopExecutor 将子容器Host提交到具体的StartChild线程类并行执行 —> … 使用​事件驱动​ 初始化 Servlet


仔细看看吧

继续 LifeCycleBase # startInternal();

调用 StandardEngine

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_start_08

@Override
protected synchronized void startInternal() throws LifecycleException {


// Standard container startup
super.startInternal();
}

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_初始化_09

父类 ContainerBase # startInternal 的方法

看下调用栈 也能看出

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_初始化_10

跟进去 关键代码

结合Server.xml配置文件 Engine节点信息

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_start_11

// 查找子容器,启动子容器  

// Start our child containers, if any
Container children[] = findChildren();
List<Future<Void>> results = new ArrayList<>();
for (Container child : children) {
results.add(startStopExecutor.submit(new StartChild(child)));
}

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_初始化_12

children事实上就是Host的集合, 然后 startStopExecutor 提交启动任务 ,这个startStopExecutor就是在load节点初始化好的,这里来使用。

那就看 StartChild 线程呗

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_start_13

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_tomcat源码_14

继续走

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_tomcat启动分析_15

跟进去 还是 LifyCycleBean --> StandardHost # startInternal -----> super.startInternal(); ----> setState(LifecycleState.STARTING);

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_初始化_16

设置生命周期 触发实例化Context —> LifyCycleBean # setStateInternal -----> fireLifecycleEvent(lifecycleEvent, data);

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_生命周期_17

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_初始化_18

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_初始化_19

触发Host的生命周期事件后,将后续工作交给生命周期监听器HostConfig来进行,Hostconfig#lifecycleEvent方法,捕获start事件,执行start方法

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_tomcat源码_20

紧接着就是下面的流程了

-------> Hostconfig#deployApps--------------> Hostconfig#deployDirectories ,以线程方式并行处理多个项目 ---------> DeployDirectory 线程类 ------>Hostconfig#deployDirectory-----------------------> 通过xml解析对象进行分析,设置一些context应用的必要属性 ,在addChild方法中完善context的过程 -------> ContainerBase#addChild方法--------------> ContainerBase#addChildInternal---------> StandardContext#startInternal方法 ------------> 给每个应用设置类加载器 ,把具体每个应用的处理交给了ContextConfig . loadOnStartup方法根据web.xml中配置servlet的load-on-startup来进行创建实例化对应servlet。执行之后,instance就有具体对象了。 ------> StandardContext#loadOnStartup方法 ------> StandardWrapper#load方法 ---------------------> StandardWrapper#loadServlet方法


Connector Start

主要流程

Connector# startInternal ------> AbstractProtocol# start -----------> AbstractEndpoint # start ----------> NioEndpoint# startInternal ----------> AbstractEndpoint#startAcceptorThreads ------> NioEndpoint# createAcceptor -------> Acceptor线程#run方法 Socket.Accept


小结

Tomcat - Tomcat 8.5.55 启动过程源码分析阶段三_start阶段_生命周期_21