Pre
上篇我们搭建了tomcat的源码环境: Tomcat - 源码构建Tomcat 8.5.55 启动
接下来我们来分析一下tomcat的启动过程。
回忆一下 tomcat的架构图
Tomcat启动的时候肯定要把架构中提到的组件进行实例化(实例化创建–>销毁等:生命周期)。
Tomcat中那么多组件,为了统一规范他们的生命周期,Tomcat抽象出了LifeCycle生命周期接口 。
生命周期统一管理组件LifeCycle
查看LifeCycle的实现类 Hierarchy, 快捷键 Ctrl + H
是不是就和刚才架构中的组件对应上了?
启动入口类查找
我们一linux操作系统为例,我们启动tomcat都是执行 startup.sh 对吧
都是shell, 不难
继续 catalina.sh start
到最后其实是执行了 java xxxx.jar org.apache.catalina.startup.Bootstrap(main) start(参数)
那就看Bootstrap的main方法吧
启动流程分析
启动总览
骨干脉络清楚了 ,看下具体实现咯
init 实例化 Bootstrap
从 Bootstrap#main() 进去 ,为了关注核心流程,精简了一下
private static volatile Bootstrap daemon = null; Bootstrap bootstrap = new Bootstrap(); bootstrap.init(); daemon = bootstrap;
init 干了啥 ?
/** * Initialize daemon. * @throws Exception Fatal initialization error */ public void init() throws Exception { // 初始化类加载器 initClassLoaders(); // 设置类加载器 Thread.currentThread().setContextClassLoader(catalinaLoader); SecurityClassLoad.securityClassLoad(catalinaLoader); // load org.apache.catalina.startup.Catalina Class<?> startupClass = catalinaLoader.loadClass("org.apache.catalina.startup.Catalina"); // 实例化Catalina Object startupInstance = startupClass.getConstructor().newInstance(); // 方法名 String methodName = "setParentClassLoader"; // 参数 Class<?> paramTypes[] = new Class[1]; paramTypes[0] = Class.forName("java.lang.ClassLoader"); Object paramValues[] = new Object[1]; paramValues[0] = sharedLoader; // 获取方法 Method method = startupInstance.getClass().getMethod(methodName, paramTypes); // 方法调用 method.invoke(startupInstance, paramValues); // 对象赋值 catalinaDaemon = startupInstance; }
是不是很熟悉? 通过反射调用 org.apache.catalina.startup.Catalina#setParentClassLoader 方法 ,最后 catalinaDaemon = startupInstance; 赋值 , 事实上 后面调用的load 和 start方法 均为Catalina对象的load和start 方法。