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

2021-08-17 11:39:30 浏览数 (1)


启动流程分析


Pre

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

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


Star阶段

start总览

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

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


start源码分析

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

还是模板模式

StandardServer Start

这里主要是继续启动Service

StandardService Start

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

一样的套路

精简后的核心代码如下:

代码语言:javascript复制
 @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

代码语言:javascript复制
  @Override
    protected synchronized void startInternal() throws LifecycleException {

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

父类 ContainerBase # startInternal 的方法

看下调用栈 也能看出

跟进去 关键代码

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

代码语言:javascript复制
// 查找子容器,启动子容器  

// 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)));
 }

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

那就看 StartChild 线程呗

继续走

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

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

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

紧接着就是下面的流程了

-------> 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


小结

0 人点赞