日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關(guān)咨詢(xún)
選擇下列產(chǎn)品馬上在線(xiàn)溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問(wèn)題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷(xiāo)解決方案
Java的ClassLoader機(jī)制解析

JVM在加載類(lèi)的時(shí)候,都是通過(guò)ClassLoader的loadClass()方法來(lái)加載class的,loadClass(String name)方法:

使用的是雙親委托模式:

jvm啟動(dòng)時(shí),會(huì)啟動(dòng)jre/rt.jar里的類(lèi)加載器:bootstrap classloader,用來(lái)加載java核心api;然后啟動(dòng)擴(kuò)展類(lèi)加載器ExtClassLoader加載擴(kuò)展類(lèi),并加載用戶(hù)程序加載器AppClassLoader,并指定ExtClassLoader為他的父類(lèi);

當(dāng)類(lèi)被加載時(shí),會(huì)先檢查在內(nèi)存中是否已經(jīng)被加載,如果是,則不再加載,如果沒(méi)有,再由AppClassLoader來(lái)加載,先從jar包里找,沒(méi)有再?gòu)腸lasspath里找;

如果自定義loader類(lèi),就會(huì)存在這命名空間的情況,不同的加載器加載同一個(gè)類(lèi)時(shí),產(chǎn)生的實(shí)例其實(shí)是不同的;

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); 

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

◆ name - 類(lèi)的二進(jìn)制名稱(chēng)

◆ resolve - 如果該參數(shù)為 true,則分析這個(gè)類(lèi)

Java代碼

 
 
 
  1. protected synchronized Class loadClass(String name, boolean resolve) 
  2. throws ClassNotFoundException 
  3. // First, check if the class has already been loaded 
  4. //JVM 規(guī)范規(guī)定ClassLoader可以在緩存保留它所加載的Class,如果一個(gè)Class已經(jīng)被加載過(guò),則直接從緩存中獲取 
  5. Class c = findLoadedClass(name); 
  6. if (c == null) { 
  7. try { 
  8. if (parent != null) { 
  9. c = parent.loadClass(name, false); 
  10. } else { 
  11. c = findBootstrapClass0(name); 
  12. } catch (ClassNotFoundException e) { 
  13. // If still not found, then invoke findClass in order 
  14. // to find the class. 
  15. c = findClass(name); 
  16. if (resolve) { 
  17. resolveClass(c); 
  18. return c; 
  19.  
  20. protected synchronized Class loadClass(String name, boolean resolve) 
  21. throws ClassNotFoundException 
  22. // First, check if the class has already been loaded 
  23. //JVM 規(guī)范規(guī)定ClassLoader可以在緩存保留它所加載的Class,如果一個(gè)Class已經(jīng)被加載過(guò),則直接從緩存中獲取 
  24. Class c = findLoadedClass(name); 
  25. if (c == null) { 
  26. try { 
  27. if (parent != null) { 
  28. c = parent.loadClass(name, false); 
  29. } else { 
  30. c = findBootstrapClass0(name); 
  31. } catch (ClassNotFoundException e) { 
  32. // If still not found, then invoke findClass in order 
  33. // to find the class. 
  34. c = findClass(name); 
  35. if (resolve) { 
  36. resolveClass(c); 
  37. return c; 

如果ClassLoader并沒(méi)有加載這個(gè)class,則調(diào)用findBootstrapClass0:

Java代碼

 
 
 
  1. private Class findBootstrapClass0(String name) 
  2. throws ClassNotFoundException 
  3. check(); 
  4. if (!checkName(name)) 
  5. throw new ClassNotFoundException(name); 
  6. return findBootstrapClass(name); 
  7.  
  8. private Class findBootstrapClass0(String name) 
  9. throws ClassNotFoundException 
  10. check(); 
  11. if (!checkName(name)) 
  12. throw new ClassNotFoundException(name); 
  13. return findBootstrapClass(name); 

該方法會(huì)調(diào)用check()方法來(lái)判斷這個(gè)類(lèi)是否已經(jīng)初始化,并且通過(guò)checkName(name)來(lái)判斷由name指定的這個(gè)類(lèi)是否存在***調(diào)用findBootstrapClass(name):

Java代碼

 
 
 
  1. private native Class findBootstrapClass(String name) 
  2. throws ClassNotFoundException; 
  3.  
  4. private native Class findBootstrapClass(String name) 
  5. throws ClassNotFoundException; 

而這個(gè)findBootstrapClass方法是一個(gè)native方法,這是我們的root loader,這個(gè)載入方法并非是由JAVA所寫(xiě),而是C++寫(xiě)的,它會(huì)最終調(diào)用JVM中的原生findBootstrapClass方法來(lái)完成類(lèi)的加載。

如果上面兩個(gè)都找不到,則使用findClass(name)來(lái)查找指定類(lèi)名的Class:

Java代碼

 
 
 
  1. protected Class findClass(String name) throws ClassNotFoundException { 
  2. throw new ClassNotFoundException(name); 
  3.  
  4. protected Class findClass(String name) throws ClassNotFoundException { 
  5. throw new ClassNotFoundException(name); 

JDK5.0中的說(shuō)明:

使用指定的二進(jìn)制名稱(chēng)查找類(lèi)。此方法應(yīng)該被類(lèi)加載器的實(shí)現(xiàn)重寫(xiě),該實(shí)現(xiàn)按照委托模型來(lái)加載類(lèi)。在通過(guò)父類(lèi)加載器檢查所請(qǐng)求的類(lèi)后,此方法將被 loadClass 方法調(diào)用。默認(rèn)實(shí)現(xiàn)拋出一個(gè)ClassNotFoundException。

所以,我們?cè)谧远x類(lèi)中,只需要重寫(xiě)findClass()即可。

MyClassLoader類(lèi):

Java代碼

 
 
 
  1. public class MyClassLoader extends ClassLoader { 
  2. private String fileName; 
  3.  
  4. public MyClassLoader(String fileName) { 
  5. this.fileName = fileName; 
  6.  
  7. protected Class findClass(String className) throws ClassNotFoundException { 
  8. Class clazz = this.findLoadedClass(className); 
  9. if (null == clazz) { 
  10. try { 
  11. String classFile = getClassFile(className); 
  12. FileInputStream fis = new FileInputStream(classFile); 
  13. FileChannel fileC = fis.getChannel(); 
  14. ByteArrayOutputStream baos = new 
  15. ByteArrayOutputStream(); 
  16. WritableByteChannel outC = Channels.newChannel(baos); 
  17. ByteBuffer buffer = ByteBuffer.allocateDirect(1024); 
  18. while (true) { 
  19. int i = fileC.read(buffer); 
  20. if (i == 0 || i == -1) { 
  21. break; 
  22. buffer.flip(); 
  23. outC.write(buffer); 
  24. buffer.clear(); 
  25. fis.close(); 
  26. byte[] bytes = baos.toByteArray(); 
  27.  
  28. clazz = defineClass(className, bytes, 0, bytes.length); 
  29. } catch (FileNotFoundException e) { 
  30. e.printStackTrace(); 
  31. } catch (IOException e) { 
  32. e.printStackTrace(); 
  33. return clazz; 
  34. private byte[] loadClassBytes(String className) throws 
  35. ClassNotFoundException { 
  36. try { 
  37. String classFile = getClassFile(className); 
  38. FileInputStream fis = new FileInputStream(classFile); 
  39. FileChannel fileC = fis.getChannel(); 
  40. ByteArrayOutputStream baos = new 
  41. ByteArrayOutputStream(); 
  42. WritableByteChannel outC = Channels.newChannel(baos); 
  43. ByteBuffer buffer = ByteBuffer.allocateDirect(1024); 
  44. while (true) { 
  45. int i = fileC.read(buffer); 
  46. if (i == 0 || i == -1) { 
  47. break; 
  48. buffer.flip(); 
  49. outC.write(buffer); 
  50. buffer.clear(); 
  51. fis.close(); 
  52. return baos.toByteArray(); 
  53. } catch (IOException fnfe) { 
  54. throw new ClassNotFoundException(className); 
  55. private String getClassFile(String name) { 
  56. StringBuffer sb = new StringBuffer(fileName); 
  57. name = name.replace('.', File.separatorChar) + ".class"; 
  58. sb.append(File.separator + name); 
  59. return sb.toString(); 
  60.  
  61. public class MyClassLoader extends ClassLoader { 
  62. private String fileName; 
  63.  
  64. public MyClassLoader(String fileName) { 
  65. this.fileName = fileName; 
  66.  
  67. protected Class findClass(String className) throws ClassNotFoundException { 
  68. Class clazz = this.findLoadedClass(className); 
  69. if (null == clazz) { 
  70. try { 
  71. String classFile = getClassFile(className); 
  72. FileInputStream fis = new FileInputStream(classFile); 
  73. FileChannel fileC = fis.getChannel(); 
  74. ByteArrayOutputStream baos = new 
  75. ByteArrayOutputStream(); 
  76. WritableByteChannel outC = Channels.newChannel(baos); 
  77. ByteBuffer buffer = ByteBuffer.allocateDirect(1024); 
  78. while (true) { 
  79. int i = fileC.read(buffer); 
  80. if (i == 0 || i == -1) { 
  81. break; 
  82. buffer.flip(); 
  83. outC.write(buffer); 
  84. buffer.clear(); 
  85. fis.close(); 
  86. byte[] bytes = baos.toByteArray(); 
  87.  
  88. clazz = defineClass(className, bytes, 0, bytes.length); 
  89. } catch (FileNotFoundException e) { 
  90. e.printStackTrace(); 
  91. } catch (IOException e) { 
  92. e.printStackTrace(); 
  93. return clazz; 
  94. private byte[] loadClassBytes(String className) throws 
  95. ClassNotFoundException { 
  96. try { 
  97. String classFile = getClassFile(className); 
  98. FileInputStream fis = new FileInputStream(classFile); 
  99. FileChannel fileC = fis.getChannel(); 
  100. ByteArrayOutputStream baos = new 
  101. ByteArrayOutputStream(); 
  102. WritableByteChannel outC = Channels.newChannel(baos); 
  103. ByteBuffer buffer = ByteBuffer.allocateDirect(1024); 
  104. while (true) { 
  105. int i = fileC.read(buffer); 
  106. if (i == 0 || i == -1) { 
  107. break; 
  108. buffer.flip(); 
  109. outC.write(buffer); 
  110. buffer.clear(); 
  111. fis.close(); 
  112. return baos.toByteArray(); 
  113. } catch (IOException fnfe) { 
  114. throw new ClassNotFoundException(className); 
  115. private String getClassFile(String name) { 
  116. StringBuffer sb = new StringBuffer(fileName); 
  117. name = name.replace('.', File.separatorChar) + ".class"; 
  118. sb.append(File.separator + name); 
  119. return sb.toString(); 

該類(lèi)中通過(guò)調(diào)用defineClass(String name, byte[] b, int off, int len)方法來(lái)定義一個(gè)類(lèi):

Java代碼

 
 
 
  1. protected final Class defineClass(String name, byte[] b, int off, int len) 
  2. throws ClassFormatError 
  3. return defineClass(name, b, off, len, null); 
  4.  
  5. protected final Class defineClass(String name, byte[] b, int off, int len) 
  6. throws ClassFormatError 
  7. return defineClass(name, b, off, len, null); 

注:MyClassLoader加載類(lèi)時(shí)有一個(gè)局限,必需指定.class文件,而不能指定.jar文件。該類(lèi)中的大部分代碼是從網(wǎng)上搜索到的,是出自一牛人之筆,只是不知道原帖在哪,希望不會(huì)被隱藏。

MainClassLoader類(lèi):

Java代碼

 
 
 
  1. public class MainClassLoader { 
  2. public static void 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. public class MainClassLoader { 
  15. public static void main(String[] args) { 
  16. try { 
  17. MyClassLoader tc = new MyClassLoader("F:\\OpenLib\\"); 
  18. Class c = tc.findClass("Test"); 
  19. c.newInstance(); 
  20. } catch (ClassNotFoundException e) { 
  21. e.printStackTrace(); 
  22. } catch (IllegalAccessException e) { 
  23. e.printStackTrace(); 
  24. } catch (InstantiationException e) { 
  25. e.printStackTrace(); 

***是一個(gè)簡(jiǎn)單的Test測(cè)試類(lèi):

Java代碼

 
 
 
  1. public class Test 
  2. public Test() { 
  3. System.out.println("Test"); 
  4. public static void main(String[] args) { 
  5. System.out.println("Hello World"); 
  6. }  

原文鏈接:http://huangcanqin.iteye.com/blog/1273170 【編輯推薦】

  1. Java內(nèi)存泄露的理解與解決
  2. 精解Java中代理模式的實(shí)現(xiàn)
  3. Java自帶的Future多線(xiàn)程模式
  4. 解析Java finally的神秘面紗
  5. 調(diào)用Java NIO提高文件讀寫(xiě)速度

本文題目:Java的ClassLoader機(jī)制解析
本文來(lái)源:http://www.dlmjj.cn/article/dhphdoe.html