亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > Java > 正文

Java中ClassLoader類加載學習總結

2024-07-13 10:14:53
字體:
來源:轉載
供稿:網友

 

雙親委派模型

類加載這個概念應該算是Java語言的一種創新,目的是為了將類的加載過程與虛擬機解耦,達到”通過類的全限定名來獲取描述此類的二進制字節流“的目的。實現這個功能的代碼模塊就是類加載器。類加載器的基本模型就是大名鼎鼎的雙親委派模型(Parents Delegation Model)。聽上去很牛掰,其實邏輯很簡單,在需要加載一個類的時候,我們首先判斷該類是否已被加載,如果沒有就判斷是否已被父加載器加載,如果還沒有再調用自己的findClass方法嘗試加載?;镜哪P途褪沁@樣(盜圖侵刪):

Java,ClassLoader,類加載

實現起來也很簡單,重點就是ClassLoader類的loadClass方法,源碼如下:

 

 

protected Class<?> loadClass(String name, boolean resolve)  throws ClassNotFoundException{  synchronized (getClassLoadingLock(name)) {    // First, check if the class has already been loaded    Class<?> c = findLoadedClass(name);    if (c == null) {      long t0 = System.nanoTime();      try {        if (parent != null) {          c = parent.loadClass(name, false);        } else {          c = findBootstrapClassOrNull(name);        }      } catch (ClassNotFoundException e) {        // ClassNotFoundException thrown if class not found        // from the non-null parent class loader      }      if (c == null) {        // If still not found, then invoke findClass in order        // to find the class.        long t1 = System.nanoTime();        c = findClass(name);        // this is the defining class loader; record the stats        sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);        sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);        sun.misc.PerfCounter.getFindClasses().increment();      }    }    if (resolve) {            Class(c);    }    return c;  }}

突然感覺被逗了,怎么默認直接就拋了異常呢?其實是因為ClassLoader這個類是一個抽象類,實際在使用時候會寫個子類,這個方法會按照需要被重寫,來完成業務需要的加載過程。

自定義ClassLoader

在自定義ClassLoader的子類時候,我們常見的會有兩種做法,一種是重寫loadClass方法,另一種是重寫findClass方法。其實這兩種方法本質上差不多,畢竟loadClass也會調用findClass,但是從邏輯上講我們最好不要直接修改loadClass的內部邏輯。

個人認為比較好的做法其實是只在findClass里重寫自定義類的加載方法。

為啥說這種比較好呢,因為前面我也說道,loadClass這個方法是實現雙親委托模型邏輯的地方,擅自修改這個方法會導致模型被破壞,容易造成問題。因此我們最好是在雙親委托模型框架內進行小范圍的改動,不破壞原有的穩定結構。同時,也避免了自己重寫loadClass方法的過程中必須寫雙親委托的重復代碼,從代碼的復用性來看,不直接修改這個方法始終是比較好的選擇。

當然,如果是刻意要破壞雙親委托模型就另說。

破壞雙親委托模型

為什么要破壞雙親委托模型呢?

其實在某些情況下,我們可能需要加載兩個不同的類,但是不巧的是這兩個類的名字完全一樣,這時候雙親委托模型就無法滿足我們的要求了,我們就要重寫loadClass方法破壞雙親委托模型,讓同一個類名加載多次。當然,這里說的破壞只是局部意義上的破壞。

但是類名相同了,jvm怎么區別這兩個類呢?顯然,這并不會造成什么世界觀的崩塌,其實類在jvm里并不僅是通過類名來限定的,他還屬于加載他的ClassLoader。由不同ClassLoader加載的類其實是互不影響的。

做一個實驗。

我們先寫兩個類:

package com.mythsman.test;public class Hello {  public void say() {    System.out.println("This is from Hello v1");  }}
package com.mythsman.test;public class Hello {  public void say() {    System.out.println("This is from Hello v2");  }}

 

兩個類名字一樣,唯一的區別是方法的實現不一樣。我們先分別編譯,然后把生成的class文件重命名為Hello.class.1和Hello.class.2。

我們的目的是希望能在測試類里分別創建這兩個類的實例。

接著我們新建一個測試類com.mythsman.test.Main,在主函數里創建兩個自定義的ClassLoader:

ClassLoader classLoader1=new ClassLoader() {  @Override  public Class<?> loadClass(String s) throws ClassNotFoundException {    try {      if (s.equals("com.mythsman.test.Hello")) {        byte[] classBytes = Files.readAllBytes(Paths.get("/home/myths/Desktop/test/Hello.class.1"));        return defineClass(s, classBytes, 0, classBytes.length);      }else{        return super.loadClass(s);      }    }catch (IOException e) {      throw new ClassNotFoundException(s);    }  }};ClassLoader classLoader2=new ClassLoader() {  @Override  public Class<?> loadClass(String s) throws ClassNotFoundException {    try {      if (s.equals("com.mythsman.test.Hello")) {        byte[] classBytes = Files.readAllBytes(Paths.get("/home/myths/Desktop/test/Hello.class.2"));        return defineClass(s, classBytes, 0, classBytes.length);      }else{        return super.loadClass(s);      }    }catch (IOException e) {      throw new ClassNotFoundException(s);    }  }};

 

這兩個ClassLoader的用途就是分別關聯Hello類的兩種不同字節碼,我們需要讀取字節碼文件并通過defineClass方法加載成class。注意我們重載的是loadClass方法,如果是重載findClass方法那么由于loadClass方法的雙親委托處理機制,第二個ClassLoader的findClass方法其實并不會被調用。

那我們怎么生成實例呢?顯然我們不能直接用類名來引用(名稱沖突),那就只能用反射了:

Object helloV1=classLoader1.loadClass("com.mythsman.test.Hello").newInstance();Object helloV2=classLoader2.loadClass("com.mythsman.test.Hello").newInstance();helloV1.getClass().getMethod("say").invoke(helloV1);helloV2.getClass().getMethod("say").invoke(helloV2);

 

輸出:

 

 

This is from Hello v1This is from Hello v2

 

OK,這樣就算是完成了兩次加載,但是還有幾個注意點需要關注下。

兩個類的關系是什么

顯然這兩個類并不是同一個類,但是他們的名字一樣,那么類似isinstance of之類的操作符結果是什么樣的呢:

System.out.println("class:"+helloV1.getClass());System.out.println("class:"+helloV2.getClass());System.out.println("hashCode:"+helloV1.getClass().hashCode());System.out.println("hashCode:"+helloV2.getClass().hashCode());System.out.println("classLoader:"+helloV1.getClass().getClassLoader());System.out.println("classLoader:"+helloV2.getClass().getClassLoader());

輸出:

 

 

class:class com.mythsman.test.Helloclass:class com.mythsman.test.HellohashCode:1581781576hashCode:1725154839classLoader:com.mythsman.test.Main$1@5e2de80cclassLoader:com.mythsman.test.Main$2@266474c2

 

他們的類名的確是一樣的,但是類的hashcode不一樣,也就意味著這兩個本質不是一個類,而且他們的類加載器也不同(其實就是Main的兩個內部類)。

這兩個類加載器跟系統的三層類加載器是什么關系

以第一個自定義的類加載器為例:

System.out.println(classLoader1.getParent().getParent().getParent());System.out.println(classLoader1.getParent().getParent());System.out.println(classLoader1.getParent());System.out.println(classLoader1 );System.out.println(ClassLoader.getSystemClassLoader());

 

輸出:

nullsun.misc.Launcher$ExtClassLoader@60e53b93sun.misc.Launcher$AppClassLoader@18b4aac2com.mythsman.test.Main$1@5e2de80csun.misc.Launcher$AppClassLoader@18b4aac2

 

 
我們可以看到,第四行就是這個自定義的ClassLoader,他的父親是AppClassLoader,爺爺是ExtClassLoader,太爺爺是null,其實就是用C寫的BootStrapClassLoader。而當前系統的ClassLoader就是這個AppClassLoader。

 

當然,這里說的父子關系并不是繼承關系,而是組合關系,子ClassLoader保存了父ClassLoader的一個引用(parent)。


注:相關教程知識閱讀請移步到JAVA教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美视频国产精品| 神马国产精品影院av| 国产成人精品久久二区二区| 在线国产精品视频| 美女撒尿一区二区三区| 国产日韩精品综合网站| 久久国产精品偷| 亲子乱一区二区三区电影| 日韩精品免费在线观看| 亚洲精品黄网在线观看| 国产精品av免费在线观看| 精品国产一区二区三区四区在线观看| 伊人久久久久久久久久久| 久久久国产精品x99av| 秋霞成人午夜鲁丝一区二区三区| 欧美一级片久久久久久久| 精品久久久久久| 一区二区三区美女xx视频| 欧美丰满少妇xxxxx做受| 日本伊人精品一区二区三区介绍| 日本人成精品视频在线| 国产精品视频资源| 亚洲日本欧美日韩高观看| 狠狠久久五月精品中文字幕| 国产精品久久久久久久久久久久| 久久久综合免费视频| 这里只有精品丝袜| 色狠狠久久aa北条麻妃| 国产精品高潮呻吟久久av野狼| 国产精品户外野外| 国产日产亚洲精品| 亚洲第一福利网站| 国产精品91一区| 3344国产精品免费看| 欧美肥婆姓交大片| 成人黄色av网站| 亚洲欧美色图片| 亚洲国产91精品在线观看| 欧美性视频精品| 九九热这里只有精品6| 久久6精品影院| 日韩av网站电影| 97在线看免费观看视频在线观看| 亚洲成人激情视频| 91社区国产高清| 国产在线观看精品| 日韩精品免费电影| 国产69精品久久久久9999| 欧美麻豆久久久久久中文| 亚洲无亚洲人成网站77777| 九九精品视频在线观看| 伊人男人综合视频网| 欧美自拍视频在线| 久久不射热爱视频精品| 国产精品日韩在线一区| 国产网站欧美日韩免费精品在线观看| 亚洲欧洲第一视频| 国产精品美女视频网站| 亚洲一区二区三区久久| 欧美中文字幕精品| 在线播放国产精品| 欧美日韩一区免费| 亚洲第一男人天堂| 亚洲人成电影在线| 欧美乱人伦中文字幕在线| 国产精品亚洲美女av网站| www.日韩视频| 国产精品第3页| 国产视频在线观看一区二区| 久久久精品一区二区三区| 欧美视频免费在线| 国产精品嫩草影院久久久| 中文字幕精品在线视频| 57pao成人永久免费视频| 欧美黑人视频一区| 91在线观看免费网站| 欧美日韩一二三四五区| 国产欧美一区二区三区视频| 高清欧美电影在线| 精品国产乱码久久久久久婷婷| 久久久久久久久久国产| 国产精品免费久久久久影院| 日韩亚洲成人av在线| 欧美日韩国产综合视频在线观看中文| 日韩美女在线观看一区| 欧美日韩综合视频网址| 伊人伊成久久人综合网站| 亚洲a一级视频| 91在线无精精品一区二区| 日韩在线精品一区| 奇米一区二区三区四区久久| 国产免费观看久久黄| 精品久久久一区| 另类天堂视频在线观看| 91精品在线看| 亚洲精品福利免费在线观看| 欧美在线观看www| 欧美激情影音先锋| 最近2019中文字幕大全第二页| 欧美日韩亚洲视频| 97久久久免费福利网址| 亚洲天堂av在线免费观看| 日韩电影在线观看免费| 国产一区二区三区欧美| 国内精久久久久久久久久人| 国产精品毛片a∨一区二区三区|国| 亚洲欧美日韩精品久久奇米色影视| 九九精品在线播放| 欧美国产日韩二区| 少妇高潮久久久久久潘金莲| 亚洲日韩中文字幕在线播放| 4438全国亚洲精品在线观看视频| 欧美激情va永久在线播放| 欧美第一淫aaasss性| 精品视频在线导航| 亚洲人在线观看| 国产精品人人做人人爽| 久久影视电视剧凤归四时歌| 亚洲毛片在线免费观看| 日韩视频在线免费| 日韩精品一区二区三区第95| 国产+人+亚洲| 亚洲天堂第二页| 国产在线观看精品一区二区三区| 亚洲精品福利免费在线观看| 国产丝袜视频一区| 91福利视频在线观看| 亚洲欧美制服另类日韩| 欧美中文在线字幕| 亚洲女人被黑人巨大进入al| 国产精品免费久久久| 日韩电影中文字幕一区| 欧美中文字幕第一页| 日韩视频免费在线观看| 亚洲日本中文字幕免费在线不卡| 成人中文字幕+乱码+中文字幕| 91国偷自产一区二区三区的观看方式| 欧美在线免费观看| 亚洲国产第一页| 久久久久北条麻妃免费看| 亚洲国产精品人久久电影| 欧美午夜激情在线| 久久久亚洲精选| 国产精品久久久久国产a级| 成人黄色免费在线观看| 91国内在线视频| 欧美一级大胆视频| 97在线视频免费观看| 亚洲一区二区福利| 国产亚洲一区精品| 欧美成人在线免费视频| 国产精品成熟老女人| 国产精品普通话| 欧美成人在线网站| 欧美人与性动交| 亚洲欧美另类中文字幕| 丝袜美腿亚洲一区二区| 日韩欧美亚洲综合| 欧美一区亚洲一区| 国产精品福利无圣光在线一区| 欧美中文字幕精品| 亚洲在线观看视频| 中文字幕精品—区二区| 欧美洲成人男女午夜视频|