JVM 詳解

ClassLoader一個常常出現又讓不少人望而卻步的詞,本文將試圖以最淺顯易懂的方式來說解 ClassLoader,但願能對不瞭解該機制的朋友起到一點點做用。

要深刻了解ClassLoader,首先就要知道ClassLoader是用來幹什麼的,顧名思義,它就是用來加載Class文件到JVM,以供程序使用的。咱們知道,java程序能夠動態加載類定義,而這個動態加載的機制就是經過ClassLoader來實現的,因此可想而知ClassLoader的重要性如何。

看到這裏,可能有的朋友會想到一個問題,那就是既然ClassLoader是用來加載類到JVM中的,那麼ClassLoader又是如何被加載呢?難道它不是java的類?

沒有錯,在這裏確實有一個ClassLoader不是用java語言所編寫的,而是JVM實現的一部分,這個ClassLoader就是bootstrap classloader(啓動類加載器),這個ClassLoader在JVM運行的時候加載java核心的API以知足java程序最基本的需求,其中就包括用戶定義的ClassLoader,這裏所謂的用戶定義是指經過java程序實現的ClassLoader,一個是ExtClassLoader,這個ClassLoader是用來加載java的擴展API的,也就是/lib/ext中的類,一個是AppClassLoader,這個ClassLoader是用來加載用戶機器上CLASSPATH設置目錄中的Class的,一般在沒有指定ClassLoader的狀況下,程序員自定義的類就由該ClassLoader進行加載。

當運行一個程序的時候,JVM啓動,運行bootstrap classloader,該ClassLoader加載java核心API(ExtClassLoader和AppClassLoader也在此時被加載),而後調用ExtClassLoader加載擴展API,最後AppClassLoader加載CLASSPATH目錄下定義的Class,這就是一個程序最基本的加載流程。

上面大概講解了一下ClassLoader的做用以及一個最基本的加載流程,接下來將講解一下ClassLoader加載的方式,這裏就不得不講一下ClassLoader在這裏使用了雙親委託模式進行類加載。

每個自定義ClassLoader都必須繼承ClassLoader這個抽象類,而每一個ClassLoader都會有一個parent ClassLoader,咱們能夠看一下ClassLoader這個抽象類中有一個getParent()方法,這個方法用來返回當前ClassLoader的parent,注意,這個parent不是指的被繼承的類,而是在實例化該ClassLoader時指定的一個ClassLoader,若是這個parent爲null,那麼就默認該ClassLoader的parent是bootstrap classloader,這個parent有什麼用呢?

咱們能夠考慮這樣一種狀況,假設咱們自定義了一個ClientDefClassLoader,咱們使用這個自定義的ClassLoader加載java.lang.String,那麼這裏String是否會被這個ClassLoader加載呢?事實上java.lang.String這個類並非被這個ClientDefClassLoader加載,而是由bootstrap classloader進行加載,爲何會這樣?實際上這就是雙親委託模式的緣由,由於在任何一個自定義ClassLoader加載一個類以前,它都會先委託它的父親ClassLoader進行加載,只有當父親ClassLoader沒法加載成功後,纔會由本身加載,在上面這個例子裏,由於java.lang.String是屬於java核心API的一個類,因此當使用ClientDefClassLoader加載它的時候,該ClassLoader會先委託它的父親ClassLoader進行加載,上面講過,當ClassLoader的parent爲null時,ClassLoader的parent就是bootstrap classloader,因此在ClassLoader的最頂層就是bootstrap classloader,所以最終委託到bootstrap classloader的時候,bootstrap classloader就會返回String的Class。

咱們來看一下ClassLoader中的一段源代碼:
java

Java代碼   收藏代碼
  1. protectedsynchronized Class loadClass(String name, boolean resolve)  程序員

  2. throws ClassNotFoundException  bootstrap

  3.   {  api

  4. // 首先檢查該name指定的class是否有被加載緩存

  5. Class c = findLoadedClass(name);  安全

  6. if (c == null) {  網絡

  7. try {  app

  8. if (parent != null) {  jvm

  9. //若是parent不爲null,則調用parent的loadClass進行加載ide

  10. = parent.loadClass(name, false);  

  11.    } else {  

  12. //parent爲null,則調用BootstrapClassLoader進行加載

  13.        c = findBootstrapClass0(name);  

  14.    }  

  15.    } catch (ClassNotFoundException e) {  

  16. //若是仍然沒法加載成功,則調用自身的findClass進行加載            

  17.        c = findClass(name);  

  18.    }  

  19. }  

  20. if (resolve) {  

  21.    resolveClass(c);  

  22. }  

  23. return c;  

  24.   }  



從上面一段代碼中,咱們能夠看出一個類加載的大概過程與以前我所舉的例子是同樣的,而咱們要實現一個自定義類的時候,只須要實現findClass方法便可。

爲何要使用這種雙親委託模式呢?

第一個緣由就是由於這樣能夠避免重複加載,當父親已經加載了該類的時候,就沒有必要子ClassLoader再加載一次。

第二個緣由就是考慮到安全因素,咱們試想一下,若是不使用這種委託模式,那咱們就能夠隨時使用自定義的String來動態替代java核心api中定義類型,這樣會存在很是大的安全隱患,而雙親委託的方式,就能夠避免這種狀況,由於String已經在啓動時被加載,因此用戶自定義類是沒法加載一個自定義的ClassLoader。

上面對ClassLoader的加載機制進行了大概的介紹,接下來不得不在此講解一下另一個和ClassLoader相關的類,那就是Class類,每一個被ClassLoader加載的class文件,最終都會以Class類的實例被程序員引用,咱們能夠把Class類看成是普通類的一個模板,JVM根據這個模板生成對應的實例,最終被程序員所使用。

咱們看到在Class類中有個靜態方法forName,這個方法和ClassLoader中的loadClass方法的目的同樣,都是用來加載class的,可是二者在做用上卻有所區別。
Class<?> loadClass(String name)
Class<?> loadClass(String name, boolean resolve)
咱們看到上面兩個方法聲明,第二個方法的第二個參數是用於設置加載類的時候是否鏈接該類,true就鏈接,不然就不鏈接。

說到鏈接,不得不在此作一下解釋,在JVM加載類的時候,須要通過三個步驟,裝載、鏈接、初始化。裝載就是找到相應的class文件,讀入JVM,初始化就不用說了,最主要就說說鏈接。

鏈接分三步,第一步是驗證class是否符合規格,第二步是準備,就是爲類變量分配內存同時設置默認初始值,第三步就是解釋,而這步就是可選的,根據上面loadClass方法的第二個參數來斷定是否須要解釋,所謂的解釋根據《深刻JVM》這本書的定義就是根據類中的符號引用查找相應的實體,再把符號引用替換成一個直接引用的過程。有點深奧吧,呵呵,在此就很少作解釋了,想具體瞭解就翻翻《深刻JVM吧》,呵呵,再這樣一步步解釋下去,那就不知道何時才能解釋得完了。

咱們再來看看那個兩個參數的loadClass方法,在JAVA API 文檔中,該方法的定義是protected,那也就是說該方法是被保護的,而用戶真正應該使用的方法是一個參數的那個,一個參數的loadclass方法實際上就是調用了兩個參數的方法,而第二個參數默認爲false,所以在這裏能夠看出經過loadClass加載類實際上就是加載的時候並不對該類進行解釋,所以也不會初始化該類。而Class類的forName方法則是相反,使用forName加載的時候就會將Class進行解釋和初始化,forName也有另一個版本的方法,能夠設置是否初始化以及設置ClassLoader,在此就很少講了。

不知道上面對這兩種加載方式的解釋是否足夠清楚,就在此舉個例子吧,例如JDBC DRIVER的加載,咱們在加載JDBC驅動的時候都是使用的forName而非是ClassLoader的loadClass方法呢?咱們知道,JDBC驅動是經過DriverManager,必須在DriverManager中註冊,若是驅動類沒有被初始化,則不能註冊到DriverManager中,所以必須使用forName而不能用loadClass。

經過ClassLoader咱們能夠自定義類加載器,定製本身所須要的加載方式,例如從網絡加載,從其餘格式的文件加載等等均可以,其實ClassLoader還有不少地方沒有講到,例如ClassLoader內部的一些實現等等,原本但願可以講得簡單易懂一點,但是結果本身看回頭好像感受並不怎麼樣,鬱悶,看來本身的文筆仍是差太多了,但願可以給一些有須要的朋友一點幫助吧。


JVM在加載類的時候,都是經過ClassLoader的loadClass()方法來加載class的,loadClass(String name)方法:

使用的是雙親委託模式:

jvm啓動時,會啓動jre/rt.jar裏的類加載器:bootstrap classloader,用來加載java核心api;而後啓動擴展類加載器ExtClassLoader加載擴展類,並加載用戶程序加載器AppClassLoader,並指定ExtClassLoader爲他的父類;

當類被加載時,會先檢查在內存中是否已經被加載,若是是,則再也不加載,若是沒有,再由AppClassLoader來加載,先從jar包裏找,沒有再從classpath裏找;

若是自定義loader類,就會存在這命名空間的狀況,不一樣的加載器加載同一個類時,產生的實例實際上是不一樣的;

Java代碼

  
  
  
  
  1. public Class<?> loadClass(String name) throws ClassNotFoundException {

  2. return loadClass(name, false);

  3. }

  4. public Class<?> loadClass(String name) throws ClassNotFoundException {

  5. return loadClass(name, false);

  6. }

loadClass(String name)方法再調用loadClass(String name, boolean resolve)方法:

◆ name - 類的二進制名稱

◆ resolve - 若是該參數爲 true,則分析這個類

Java代碼

  
  
  
  
  1. protectedsynchronized Class<?> loadClass(String name, boolean resolve)

  2. throws ClassNotFoundException

  3. {

  4. // First, check if the class has already been loaded

  5. //JVM 規範規定ClassLoader能夠在緩存保留它所加載的Class,若是一個Class已經被加載過,則直接從緩存中獲取

  6. Class c = findLoadedClass(name);

  7. if (c == null) {

  8. try {

  9. if (parent != null) {

  10. c = parent.loadClass(name, false);

  11. } else {

  12. c = findBootstrapClass0(name);

  13. }

  14. } catch (ClassNotFoundException e) {

  15. // If still not found, then invoke findClass in order

  16. // to find the class.

  17. c = findClass(name);

  18. }

  19. }

  20. if (resolve) {

  21. resolveClass(c);

  22. }

  23. return c;

  24. }

  25. protectedsynchronized Class<?> loadClass(String name, boolean resolve)

  26. throws ClassNotFoundException

  27. {

  28. // First, check if the class has already been loaded

  29. //JVM 規範規定ClassLoader能夠在緩存保留它所加載的Class,若是一個Class已經被加載過,則直接從緩存中獲取

  30. Class c = findLoadedClass(name);

  31. if (c == null) {

  32. try {

  33. if (parent != null) {

  34. c = parent.loadClass(name, false);

  35. } else {

  36. c = findBootstrapClass0(name);

  37. }

  38. } catch (ClassNotFoundException e) {

  39. // If still not found, then invoke findClass in order

  40. // to find the class.

  41. c = findClass(name);

  42. }

  43. }

  44. if (resolve) {

  45. resolveClass(c);

  46. }

  47. return c;

  48. }

若是ClassLoader並無加載這個class,則調用findBootstrapClass0:

Java代碼

  
  
  
  
  1. private Class findBootstrapClass0(String name)

  2. throws ClassNotFoundException

  3. {

  4. check();

  5. if (!checkName(name))

  6. thrownew ClassNotFoundException(name);

  7. return findBootstrapClass(name);

  8. }

  9. private Class findBootstrapClass0(String name)

  10. throws ClassNotFoundException

  11. {

  12. check();

  13. if (!checkName(name))

  14. thrownew ClassNotFoundException(name);

  15. return findBootstrapClass(name);

  16. }

該方法會調用check()方法來判斷這個類是否已經初始化,而且經過checkName(name)來判斷由name指定的這個類是否存在最後調用findBootstrapClass(name):

Java代碼

  
  
  
  
  1. privatenative Class findBootstrapClass(String name)

  2. throws ClassNotFoundException;

  3. privatenative Class findBootstrapClass(String name)

  4. throws ClassNotFoundException;

而這個findBootstrapClass方法是一個native方法,這是咱們的root loader,這個載入方法並不是是由JAVA所寫,而是C++寫的,它會最終調用JVM中的原生findBootstrapClass方法來完成類的加載。

若是上面兩個都找不到,則使用findClass(name)來查找指定類名的Class:

Java代碼

  
  
  
  
  1. protected Class<?> findClass(String name) throws ClassNotFoundException {

  2. thrownew ClassNotFoundException(name);

  3. }

  4. protected Class<?> findClass(String name) throws ClassNotFoundException {

  5. thrownew ClassNotFoundException(name);

  6. }

JDK5.0中的說明:

使用指定的二進制名稱查找類。此方法應該被類加載器的實現重寫,該實現按照委託模型來加載類。在經過父類加載器檢查所請求的類後,此方法將被 loadClass 方法調用。默認實現拋出一個ClassNotFoundException。

因此,咱們在自定義類中,只須要重寫findClass()便可。

MyClassLoader類:

Java代碼

  
  
  
  
  1. publicclass MyClassLoader extends ClassLoader {

  2. private String fileName;

  3. public MyClassLoader(String fileName) {

  4. this.fileName = fileName;

  5. }

  6. protected Class<?> findClass(String className) throws ClassNotFoundException {

  7. Class clazz = this.findLoadedClass(className);

  8. if (null == clazz) {

  9. try {

  10. String classFile = getClassFile(className);

  11. FileInputStream fis = new FileInputStream(classFile);

  12. FileChannel fileC = fis.getChannel();

  13. ByteArrayOutputStream baos = new

  14. ByteArrayOutputStream();

  15. WritableByteChannel outC = Channels.newChannel(baos);

  16. ByteBuffer buffer = ByteBuffer.allocateDirect(1024);

  17. while (true) {

  18. int i = fileC.read(buffer);

  19. if (i == 0 || i == -1) {

  20. break;

  21. }

  22. buffer.flip();

  23. outC.write(buffer);

  24. buffer.clear();

  25. }

  26. fis.close();

  27. byte[] bytes = baos.toByteArray();

  28. clazz = defineClass(className, bytes, 0, bytes.length);

  29. } catch (FileNotFoundException e) {

  30. e.printStackTrace();

  31. } catch (IOException e) {

  32. e.printStackTrace();

  33. }

  34. }

  35. return clazz;

  36. }

  37. privatebyte[] loadClassBytes(String className) throws

  38. ClassNotFoundException {

  39. try {

  40. String classFile = getClassFile(className);

  41. FileInputStream fis = new FileInputStream(classFile);

  42. FileChannel fileC = fis.getChannel();

  43. ByteArrayOutputStream baos = new

  44. ByteArrayOutputStream();

  45. WritableByteChannel outC = Channels.newChannel(baos);

  46. ByteBuffer buffer = ByteBuffer.allocateDirect(1024);

  47. while (true) {

  48. int i = fileC.read(buffer);

  49. if (i == 0 || i == -1) {

  50. break;

  51. }

  52. buffer.flip();

  53. outC.write(buffer);

  54. buffer.clear();

  55. }

  56. fis.close();

  57. return baos.toByteArray();

  58. } catch (IOException fnfe) {

  59. thrownew ClassNotFoundException(className);

  60. }

  61. }

  62. private String getClassFile(String name) {

  63. StringBuffer sb = new StringBuffer(fileName);

  64. name = name.replace('.', File.separatorChar) + ".class";

  65. sb.append(File.separator + name);

  66. return sb.toString();

  67. }

  68. }

  69. publicclass MyClassLoader extends ClassLoader {

  70. private String fileName;

  71. public MyClassLoader(String fileName) {

  72. this.fileName = fileName;

  73. }

  74. protected Class<?> findClass(String className) throws ClassNotFoundException {

  75. Class clazz = this.findLoadedClass(className);

  76. if (null == clazz) {

  77. try {

  78. String classFile = getClassFile(className);

  79. FileInputStream fis = new FileInputStream(classFile);

  80. FileChannel fileC = fis.getChannel();

  81. ByteArrayOutputStream baos = new

  82. ByteArrayOutputStream();

  83. WritableByteChannel outC = Channels.newChannel(baos);

  84. ByteBuffer buffer = ByteBuffer.allocateDirect(1024);

  85. while (true) {

  86. int i = fileC.read(buffer);

  87. if (i == 0 || i == -1) {

  88. break;

  89. }

  90. buffer.flip();

  91. outC.write(buffer);

  92. buffer.clear();

  93. }

  94. fis.close();

  95. byte[] bytes = baos.toByteArray();

  96. clazz = defineClass(className, bytes, 0, bytes.length);

  97. } catch (FileNotFoundException e) {

  98. e.printStackTrace();

  99. } catch (IOException e) {

  100. e.printStackTrace();

  101. }

  102. }

  103. return clazz;

  104. }

  105. privatebyte[] loadClassBytes(String className) throws

  106. ClassNotFoundException {

  107. try {

  108. String classFile = getClassFile(className);

  109. FileInputStream fis = new FileInputStream(classFile);

  110. FileChannel fileC = fis.getChannel();

  111. ByteArrayOutputStream baos = new

  112. ByteArrayOutputStream();

  113. WritableByteChannel outC = Channels.newChannel(baos);

  114. ByteBuffer buffer = ByteBuffer.allocateDirect(1024);

  115. while (true) {

  116. int i = fileC.read(buffer);

  117. if (i == 0 || i == -1) {

  118. break;

  119. }

  120. buffer.flip();

  121. outC.write(buffer);

  122. buffer.clear();

  123. }

  124. fis.close();

  125. return baos.toByteArray();

  126. } catch (IOException fnfe) {

  127. thrownew ClassNotFoundException(className);

  128. }

  129. }

  130. private String getClassFile(String name) {

  131. StringBuffer sb = new StringBuffer(fileName);

  132. name = name.replace('.', File.separatorChar) + ".class";

  133. sb.append(File.separator + name);

  134. return sb.toString();

  135. }

  136. }

該類中經過調用defineClass(String name, byte[] b, int off, int len)方法來定義一個類:

Java代碼

  
  
  
  
  1. protectedfinal Class<?> defineClass(String name, byte[] b, int off, int len)

  2. throws ClassFormatError

  3. {

  4. return defineClass(name, b, off, len, null);

  5. }

  6. protectedfinal Class<?> defineClass(String name, byte[] b, int off, int len)

  7. throws ClassFormatError

  8. {

  9. return defineClass(name, b, off, len, null);

  10. }

注:MyClassLoader加載類時有一個侷限,必需指定.class文件,而不能指定.jar文件。該類中的大部分代碼是從網上搜索到的,是出自一牛人之筆,只是不知道原帖在哪,但願不會被隱藏。

MainClassLoader類:

Java代碼

  
  
  
  
  1. publicclass MainClassLoader {

  2. publicstaticvoid main(String[] args) {

  3. try {

  4. MyClassLoader tc = new MyClassLoader("F:\\OpenLib\\");

  5. Class c = tc.findClass("Test");

  6. c.newInstance();

  7. } catch (ClassNotFoundException e) {

  8. e.printStackTrace();

  9. } catch (IllegalAccessException e) {

  10. e.printStackTrace();

  11. } catch (InstantiationException e) {

  12. e.printStackTrace();

  13. }

  14. }

  15. }

  16. publicclass MainClassLoader {

  17. publicstaticvoid main(String[] args) {

  18. try {

  19. MyClassLoader tc = new MyClassLoader("F:\\OpenLib\\");

  20. Class c = tc.findClass("Test");

  21. c.newInstance();

  22. } catch (ClassNotFoundException e) {

  23. e.printStackTrace();

  24. } catch (IllegalAccessException e) {

  25. e.printStackTrace();

  26. } catch (InstantiationException e) {

  27. e.printStackTrace();

  28. }

  29. }

  30. }

最後是一個簡單的Test測試類:

Java代碼

  
  
  
  
  1. publicclass Test

  2. {

  3. public Test() {

  4. System.out.println("Test");

  5. }

  6. publicstaticvoid main(String[] args) {

  7. System.out.println("Hello World");

  8. }

  9. }  

相關文章
相關標籤/搜索