黄色网址大全免费-黄色网址你懂得-黄色网址你懂的-黄色网址有那些-免费超爽视频-免费大片黄国产在线观看

專注Java教育14年 全國咨詢/投訴熱線:400-8080-105
動力節(jié)點LOGO圖
始于2009,口口相傳的Java黃埔軍校
首頁 hot資訊 Tomcat類加載機制

Tomcat類加載機制

更新時間:2020-12-03 17:16:48 來源:動力節(jié)點 瀏覽1503次

說到Tomcat類加載機制,我們不得不提及JVM的類加載,然后Tomcat也是運行在JVM上的。所以,我們先拋磚引玉,一起來看看JVM類加載。

JVM類加載采用父類委托機制,當JVM運行過程中,用戶需要加載某些類時,用戶自己的類加載器,把加載請求傳給父加載器,父加載器再傳給其父加載器,一直到加載器樹的頂層。最頂層的類加載器首先針對其特定的位置加載,如果加載不到就轉(zhuǎn)交給子類。如果一直到底層的類加載都沒有加載到,那么就會拋出異常ClassNotFoundException。

回過頭來,我們再來看Tomcat類加載,Tomcat類加載機制是違反了雙親委托原則的,對于一些未加載的非基礎類(Object,String等),各個web應用自己的類加載器(WebAppClassLoader)會優(yōu)先加載,加載不到時再交給commonClassLoader走雙親委托。

下面的簡圖是Tomcat9版本的官方文檔給出的Tomcat的類加載器的圖。

????Bootstrap
??????????|
???????System
??????????|
???????Common
???????/????\
??Webapp1 ??Webapp2 ..

Bootstrap :是Java的最高的加載器,用C語言實現(xiàn),主要用來加載JVM啟動時所需要的核心類,例如$JAVA_HOME/jre/lib/ext路徑下的類。

System: 會加載CLASSPATH系統(tǒng)變量所定義路徑的所有的類。

Common:會加載Tomcat路徑下的lib文件下的所有類。

Webapp1、Webapp2……: 會加載webapp路徑下項目中的所有的類。一個項目對應一個WebappClassLoader,這樣就實現(xiàn)了應用之間類的隔離了。

這3個部分,在上面的Java雙親委派模型圖中都有體現(xiàn)。不過可以看到ExtClassLoader沒有畫出來,可以理解為是跟bootstrap合并了,都是去JAVA_HOME/jre/lib下面加載類。那么Tomcat為什么要自定義類加載器呢?

隔離不同應用:部署在同一個Tomcat中的不同應用A和B,例如A用了Spring2.5。B用了Spring3.5,那么這兩個應用如果使用的是同一個類加載器,那么Web應用就會因為jar包覆蓋而無法啟動。

靈活性:Web應用之間的類加載器相互獨立,那么就可以根據(jù)修改不同的文件重建不同的類加載器替換原來的。從而不影響其他應用。

性能:如果在一個Tomcat部署多個應用,多個應用中都有相同的類庫依賴。那么可以把這相同的類庫讓Common類加載器進行加載。

Tomcat自定義了WebAppClassLoader類加載器。打破了雙親委派的機制,即如果收到類加載的請求,會嘗試自己去加載,如果找不到再交給父加載器去加載,目的就是為了優(yōu)先加載Web應用自己定義的類。我們知道ClassLoader默認的loadClass方法是以雙親委派的模型進行加載類的,那么Tomcat既然要打破這個規(guī)則,就要重寫loadClass方法,我們可以看WebAppClassLoader類中重寫的loadClass方法。

@Override
public Class loadClass(String name, boolean resolve) throws ClassNotFoundException {
????synchronized (getClassLoadingLock(name)) {
????????Class clazz = null;
// 1. 從本地緩存中查找是否加載過此類
????????clazz = findLoadedClass0(name);
????????if (clazz != null) {
????????????if (log.isDebugEnabled())
????????????????log.debug(" ?Returning class from cache");
????????????if (resolve)
????????????????resolveClass(clazz);
????????????return clazz;
????????}

// 2. 從AppClassLoader中查找是否加載過此類
????????clazz = findLoadedClass(name);
????????if (clazz != null) {
????????????if (log.isDebugEnabled())
????????????????log.debug(" ?Returning class from cache");
????????????if (resolve)
????????????????resolveClass(clazz);
????????????return clazz;
????????}

????????String resourceName = binaryNameToPath(name, false);
// 3. 嘗試用ExtClassLoader 類加載器加載類,防止Web應用覆蓋JRE的核心類
????????ClassLoader javaseLoader = getJavaseClassLoader();
????????boolean tryLoadingFromJavaseLoader;
????????try {
????????????URL url;
????????????if (securityManager != null) {
????????????????PrivilegedActiondp = new PrivilegedJavaseGetResource(resourceName);
????????????????url = AccessController.doPrivileged(dp);
????????????} else {
????????????????url = javaseLoader.getResource(resourceName);
????????????}
????????????tryLoadingFromJavaseLoader = (url != null);
????????} catch (Throwable t) {
????????????tryLoadingFromJavaseLoader = true;
????????}

????????boolean delegateLoad = delegate || filter(name, true);
?// 4. 判斷是否設置了delegate屬性,如果設置為true那么就按照雙親委派機制加載類
????????if (delegateLoad) {
????????????if (log.isDebugEnabled())
????????????????log.debug(" ?Delegating to parent classloader1 " + parent);
????????????try {
????????????????clazz = Class.forName(name, false, parent);
????????????????if (clazz != null) {
????????????????????if (log.isDebugEnabled())
????????????????????????log.debug(" ?Loading class from parent");
????????????????????if (resolve)
????????????????????????resolveClass(clazz);
????????????????????return clazz;
????????????????}
????????????} catch (ClassNotFoundException e) {
????????????????// Ignore
????????????}
????????}
? // 5. 默認是設置delegate是false的,那么就會先用WebAppClassLoader進行加載
????????if (log.isDebugEnabled())
????????????log.debug(" ?Searching local repositories");
????????try {
????????????clazz = findClass(name);
????????????if (clazz != null) {
????????????????if (log.isDebugEnabled())
????????????????????log.debug(" ?Loading class from local repository");
????????????????if (resolve)
????????????????????resolveClass(clazz);
????????????????return clazz;
????????????}
????????} catch (ClassNotFoundException e) {
????????????// Ignore
????????}

?// 6. 如果此時在WebAppClassLoader沒找到類,那么就委托給AppClassLoader去加載
????????if (!delegateLoad) {
????????????if (log.isDebugEnabled())
????????????????log.debug(" ?Delegating to parent classloader at end: " + parent);
????????????try {
????????????????clazz = Class.forName(name, false, parent);
????????????????if (clazz != null) {
????????????????????if (log.isDebugEnabled())
????????????????????????log.debug(" ?Loading class from parent");
????????????????????if (resolve)
????????????????????????resolveClass(clazz);
????????????????????return clazz;
????????????????}
????????????} catch (ClassNotFoundException e) {
????????????????// Ignore
????????????}
????????}
????}
????throw new ClassNotFoundException(name);
}

我們總結(jié)起來就是Web應用默認的類加載順序是(打破了雙親委派規(guī)則):

1.先從JVM的BootStrapClassLoader中加載。

2.加載Web應用下/WEB-INF/classes中的類。

3.加載Web應用下/WEB-INF/lib/*.jap中的jar包中的類。

4.加載上面定義的System路徑下面的類。

5.加載上面定義的Common路徑下面的類。

到此為止,Tomcat類加載機制逐漸明朗,Tomcat類加載機制的重點就是打破了雙親委派機制,WebAppClassLoader加載類的時候,繞開 AppClassLoader,直接先使用 ExtClassLoader 來加載類。這不僅保證了基礎類不會被同時加載,也

保證了在同一個 Tomcat 下不同 web 之間的 class 是相互隔離的。好了。Tomcat的類加載機制就講到這里,感興趣的小伙伴可以去觀看本站的Tomcat服務器教程,深入學習Tomcat里的各種技術。

提交申請后,顧問老師會電話與您溝通安排學習

免費課程推薦 >>
技術文檔推薦 >>
主站蜘蛛池模板: 久久久无码精品亚洲日韩按摩 | 免费看h成年漫画在线观看 免费看a级 | 久久青草免费97线频观 | 狠狠色噜噜狠狠狠合久 | 成人看片软件 | 一级毛片观看 | 97视频精品全国在线观看 | 天天躁天天爽 | 日日摸夜夜添夜夜添特色大片 | 成年人网站在线免费观看 | 日韩在线视频精品 | 欧美一区二区三区不卡视频 | 欧美性高清video | 波多野结衣xxxx性精品 | 国产乱人免费视频 | 国产精品天干天干在线综合 | a在线观看欧美在线观看 | 曰批免费视频播放免费观看网站 | 免费伦理片网站 | 99爱在线精品视频免费观看9 | 亚洲欧美日韩高清中文在线 | 精品视频二区 | 天天射天天操天天 | 91免费国产高清在线 | 亚洲人成在线观看一区二区 | 成人av手机在线观看 | 久久精品国产半推半就 | 中文字幕日本不卡一二三区 | 成人精品一区二区三区 | 色婷婷激婷婷深爱五月小说 | 性视频在线 | 天天摸日日操 | 算你色永久免费视频播放 | 日本波多野结衣在线 | 一区二区三区四区国产 | 国产成人精品综合在线 | 黄色大片在线看 | 日韩中文字幕免费在线观看 | 欧美高清hd18日本 | 国产v在线在线观看羞羞答答 | 亚洲天堂久久 |