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() 进去 ,为了关注核心流程,精简了一下
代码语言:javascript复制 private static volatile Bootstrap daemon = null;
Bootstrap bootstrap = new Bootstrap();
bootstrap.init();
daemon = bootstrap;
init 干了啥 ?
代码语言:javascript复制 /**
* 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 方法。