Tomcat啟動(dòng)核心流程示例詳解
目錄
- 一、Tomcat的啟動(dòng)核心流程
- 1.啟動(dòng)的入口
- 2.init方法
- 3.load方法
- 4.start方法
- 5.核心流程的總結(jié)
一、Tomcat的啟動(dòng)核心流程
前面給大家介紹了Tomcat中的生命周期的設(shè)計(jì),掌握了這塊對(duì)于我們分析Tomcat的核心流程是非常有幫助的,也就是我們需要?jiǎng)?chuàng)建相關(guān)的核心組件,比如Server,Service肯定都繞不開(kāi)生命周期的方法。
1.啟動(dòng)的入口
你可以通過(guò)腳本來(lái)啟動(dòng)Tomcat服務(wù)(startup.bat),但如果你看過(guò)腳本的命令,你會(huì)發(fā)現(xiàn)最終調(diào)用的還是Bootstrap中的main方法,所以我們需要從main方法來(lái)開(kāi)始
然后我們?nèi)タ磎ain方法中的代碼,我們需要重點(diǎn)關(guān)注的方法有三個(gè)
- bootstrap.init()方法
- load()方法
- start()方法
也就是在這三個(gè)方法中會(huì)完成Tomcat的核心操作。
2.init方法
我們來(lái)看下init方法中的代碼,非核心的我們直接去掉
public void init() throws Exception {// 創(chuàng)建相關(guān)的類(lèi)加載器initClassLoaders();// 省略部分代碼...// 通過(guò)反射創(chuàng)建了 Catalina 類(lèi)對(duì)象Class<?> startupClass = catalinaLoader .loadClass("org.apache.catalina.startup.Catalina");// 創(chuàng)建了 Catalina 實(shí)例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 實(shí)例 賦值給了 catalinaDaemoncatalinaDaemon = startupInstance; }
- 首先是調(diào)用了initClassLoaders()方法,這個(gè)方法會(huì)完成對(duì)應(yīng)的ClassLoader的創(chuàng)建,這個(gè)比較重要,后面專(zhuān)門(mén)寫(xiě)一篇文章來(lái)介紹。
- 通過(guò)反射的方式創(chuàng)建了Catalina的類(lèi)對(duì)象,并通過(guò)反射創(chuàng)建了Catalina的實(shí)例
- 設(shè)置了類(lèi)加載器的父子關(guān)系
- 用過(guò)成員變量catalinaDaemon記錄了我們創(chuàng)建的Catalina實(shí)例
這個(gè)是通過(guò)bootstrap.init()方法我們可以獲取到的有用的信息。然后我們繼續(xù)往下面看。
3.load方法
然后我們來(lái)看下load方法做了什么事情,代碼如下:
private void load(String[] arguments) throws Exception {// Call the load() methodString methodName = "load"; // load方法的名稱(chēng)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 對(duì)象Method method = catalinaDaemon.getClass().getMethod(methodName, paramTypes);if (log.isDebugEnabled()) { log.debug("Calling startup class " + method);}// 會(huì)執(zhí)行 Catalina的load方法method.invoke(catalinaDaemon, param); }
上面的代碼非常簡(jiǎn)單,通過(guò)注釋我們也可以看出該方法的作用是調(diào)用 Catalina的load方法。所以我們還需要加入到Catalina的load方法中來(lái)查看,代碼同樣比較長(zhǎng),只留下關(guān)鍵代碼
public void load() {if (loaded) { return; // 只能被加載一次}loaded = true;initDirs(); // 廢棄的方法// Before digester - it may be neededinitNaming(); // 和JNDI 相關(guān)的內(nèi)容 忽略// Create and execute our Digester// 創(chuàng)建并且執(zhí)行我們的 Digester 對(duì)象 Server.xmlDigester digester = createStartDigester();// 省略掉了 Digester文件處理的代碼getServer().setCatalina(this); // Server對(duì)象綁定 Catalina對(duì)象getServer().setCatalinaHome(Bootstrap.getCatalinaHomeFile());getServer().setCatalinaBase(Bootstrap.getCatalinaBaseFile());// Stream redirectioninitStreams();// 省略掉了部分代碼... getServer().init(); // 完成 Server Service Engine Connector等組件的init操作 }
把上面的代碼簡(jiǎn)化后我們發(fā)現(xiàn)這個(gè)Load方法其實(shí)也是蠻簡(jiǎn)單的,就做了兩件事。
- 通過(guò)Apache下的Digester組件完成了Server.xml文件的解析
- 通過(guò)getServer().init() 方法完成了Server,Service,Engin,Connector等核心組件的初始化操作,這塊和前面的LifecycleBase呼應(yīng)起來(lái)了。
如果生命周期的內(nèi)容不清楚,請(qǐng)看前面內(nèi)容介紹
4.start方法
最后我們來(lái)看下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); }
上面的代碼邏輯也很清楚,就是通過(guò)反射的方式調(diào)用了Catalina對(duì)象的start方法。所以進(jìn)入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)鍵方法--->啟動(dòng)Servertry { getServer().start();} catch (LifecycleException e) { // 省略...}// 省略...// Register shutdown hook 注冊(cè)關(guān)閉的鉤子if (useShutdownHook) { // 省略...}if (await) { await(); stop();} }
通過(guò)上面的代碼我們可以發(fā)現(xiàn)核心的代碼還是getServer.start()方法,也就是通過(guò)Server對(duì)象來(lái)嵌套的調(diào)用相關(guān)注解的start方法。
5.核心流程的總結(jié)
我們可以通過(guò)下圖來(lái)總結(jié)下Tomcat啟動(dòng)的核心流程
從圖中我們可以看到Bootstrap其實(shí)沒(méi)有做什么核心的事情,主要還是Catalina來(lái)完成的。
以上就是Tomcat啟動(dòng)核心流程示例詳解的詳細(xì)內(nèi)容,更多關(guān)于Tomcat啟動(dòng)流程的資料請(qǐng)關(guān)注其它相關(guān)文章!
相關(guān)文章:
1. 傳統(tǒng)tomcat啟動(dòng)服務(wù)與springboot啟動(dòng)內(nèi)置tomcat服務(wù)的區(qū)別(推薦)2. tomcat啟動(dòng)異常:子容器啟動(dòng)失敗(a child container failed during start)3. Tomcat啟動(dòng)成功但無(wú)法訪問(wèn)http://localhost:8080/的解決方法4. 為什么我的tomcat啟動(dòng)不起來(lái)5. 解決tomcat啟動(dòng) ssm項(xiàng)目出現(xiàn)亂碼的問(wèn)題6. 解讀Tomcat啟動(dòng)、重啟、暫停操作(window)7. 關(guān)于Tomcat啟動(dòng)失敗報(bào)循環(huán)依賴(lài)的問(wèn)題解決(AncestorAxisIterator)8. Tomcat啟動(dòng)報(bào)錯(cuò):嚴(yán)重: Unable to process Jar entry [module-info.class]9. 解決Tomcat啟動(dòng)失敗:嚴(yán)重 [main] org.apache.catalina.util.LifecycleBase.handleSubClassException 初始化組件失敗
