網(wǎng)站首頁 編程語言 正文
一、Tomcat的啟動核心流程
前面給大家介紹了Tomcat中的生命周期的設(shè)計,掌握了這塊對于我們分析Tomcat的核心流程是非常有幫助的,也就是我們需要創(chuàng)建相關(guān)的核心組件,比如Server,Service肯定都繞不開生命周期的方法。
1.啟動的入口
你可以通過腳本來啟動Tomcat服務(wù)(startup.bat),但如果你看過腳本的命令,你會發(fā)現(xiàn)最終調(diào)用的還是Bootstrap中的main方法,所以我們需要從main方法來開始
然后我們?nèi)タ磎ain方法中的代碼,我們需要重點關(guān)注的方法有三個
- bootstrap.init()方法
- load()方法
- start()方法
也就是在這三個方法中會完成Tomcat的核心操作。
2.init方法
我們來看下init方法中的代碼,非核心的我們直接去掉
public void init() throws Exception {
// 創(chuàng)建相關(guān)的類加載器
initClassLoaders();
// 省略部分代碼...
// 通過反射創(chuàng)建了 Catalina 類對象
Class<?> startupClass = catalinaLoader
.loadClass("org.apache.catalina.startup.Catalina");
// 創(chuàng)建了 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;
// 把 sharedLoader 設(shè)置為了 commonLoader的父加載器
Method method =
startupInstance.getClass().getMethod(methodName, paramTypes);
method.invoke(startupInstance, paramValues);
// Catalina 實例 賦值給了 catalinaDaemon
catalinaDaemon = startupInstance;
}
- 首先是調(diào)用了initClassLoaders()方法,這個方法會完成對應(yīng)的ClassLoader的創(chuàng)建,這個比較重要,后面專門寫一篇文章來介紹。
- 通過反射的方式創(chuàng)建了Catalina的類對象,并通過反射創(chuàng)建了Catalina的實例
- 設(shè)置了類加載器的父子關(guān)系
- 用過成員變量catalinaDaemon記錄了我們創(chuàng)建的Catalina實例
這個是通過bootstrap.init()方法我們可以獲取到的有用的信息。然后我們繼續(xù)往下面看。
3.load方法
然后我們來看下load方法做了什么事情,代碼如下:
private void load(String[] arguments) throws Exception {
// Call the load() method
String methodName = "load"; // load方法的名稱
Object param[];
Class<?> paramTypes[];
if (arguments==null || arguments.length==0) {
paramTypes = null;
param = null;
} else {
paramTypes = new Class[1];
paramTypes[0] = arguments.getClass();
param = new Object[1];
param[0] = arguments;
}
// catalinaDaemon 就是在 init中創(chuàng)建的 Catalina 對象
Method method =
catalinaDaemon.getClass().getMethod(methodName, paramTypes);
if (log.isDebugEnabled()) {
log.debug("Calling startup class " + method);
}
// 會執(zhí)行 Catalina的load方法
method.invoke(catalinaDaemon, param);
}
上面的代碼非常簡單,通過注釋我們也可以看出該方法的作用是調(diào)用 Catalina的load方法。所以我們還需要加入到Catalina的load方法中來查看,代碼同樣比較長,只留下關(guān)鍵代碼
public void load() {
if (loaded) {
return; // 只能被加載一次
}
loaded = true;
initDirs(); // 廢棄的方法
// Before digester - it may be needed
initNaming(); // 和JNDI 相關(guān)的內(nèi)容 忽略
// Create and execute our Digester
// 創(chuàng)建并且執(zhí)行我們的 Digester 對象 Server.xml
Digester digester = createStartDigester();
// 省略掉了 Digester文件處理的代碼
getServer().setCatalina(this); // Server對象綁定 Catalina對象
getServer().setCatalinaHome(Bootstrap.getCatalinaHomeFile());
getServer().setCatalinaBase(Bootstrap.getCatalinaBaseFile());
// Stream redirection
initStreams();
// 省略掉了部分代碼...
getServer().init(); // 完成 Server Service Engine Connector等組件的init操作
}
把上面的代碼簡化后我們發(fā)現(xiàn)這個Load方法其實也是蠻簡單的,就做了兩件事。
- 通過Apache下的Digester組件完成了Server.xml文件的解析
- 通過getServer().init() 方法完成了Server,Service,Engin,Connector等核心組件的初始化操作,這塊和前面的LifecycleBase呼應(yīng)起來了。
如果生命周期的內(nèi)容不清楚,請看前面內(nèi)容介紹
4.start方法
最后我們來看下start方法的代碼。
public void start() throws Exception {
if (catalinaDaemon == null) {
init(); // 如果 catalinaDaemon 為空 初始化操作
}
// 獲取的是 Catalina 中的 start方法
Method method = catalinaDaemon.getClass().getMethod("start", (Class [])null);
// 執(zhí)行 Catalina 的start方法
method.invoke(catalinaDaemon, (Object [])null);
}
上面的代碼邏輯也很清楚,就是通過反射的方式調(diào)用了Catalina對象的start方法。所以進入Catalina的start方法中查看。
public void start() {
if (getServer() == null) {
load(); // 如果Server 為空 重新 init 相關(guān)的組件
}
if (getServer() == null) {
log.fatal("Cannot start server. Server instance is not configured.");
return;
}
// Start the new server 關(guān)鍵方法--->啟動Server
try {
getServer().start();
} catch (LifecycleException e) {
// 省略...
}
// 省略...
// Register shutdown hook 注冊關(guān)閉的鉤子
if (useShutdownHook) {
// 省略...
}
if (await) {
await();
stop();
}
}
通過上面的代碼我們可以發(fā)現(xiàn)核心的代碼還是getServer.start()方法,也就是通過Server對象來嵌套的調(diào)用相關(guān)注解的start方法。
5.核心流程的總結(jié)
我們可以通過下圖來總結(jié)下Tomcat啟動的核心流程
從圖中我們可以看到Bootstrap其實沒有做什么核心的事情,主要還是Catalina來完成的。
原文鏈接:https://juejin.cn/post/7132710433475002405
相關(guān)推薦
- 2024-01-06 RocketMQ消費模式
- 2023-10-15 動態(tài)演示操作系統(tǒng)進程調(diào)度算法,F(xiàn)CFS, RR, SPN, SRT, HRRN
- 2022-08-06 Python導(dǎo)入不同文件夾中文件的方法詳解_python
- 2022-04-18 python字符串格式化(%格式符和format方式)_python
- 2022-04-01 Python 中 __name__ == '__main__' 的作用
- 2022-09-06 Go結(jié)構(gòu)體SliceHeader及StringHeader作用詳解_Golang
- 2022-12-10 C++?Boost?Spirit精通教程_C 語言
- 2022-04-26 python計算寄送包裹重量的實現(xiàn)過程_python
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支