Java的ClassLoader机制解析

开发 后端
类加载就是寻找一个类或是一个接口的字节码文件并通过解析该字节码来构造代表这个类或是这个接口的 class 对象的过程,下面我们将来了解一下Java的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); 

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

◆ name - 类的二进制名称

◆ resolve - 如果该参数为 true,则分析这个类

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 规范规定ClassLoader可以在缓存保留它所加载的Class,如果一个Class已经被加载过,则直接从缓存中获取 
  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 规范规定ClassLoader可以在缓存保留它所加载的Class,如果一个Class已经被加载过,则直接从缓存中获取 
  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并没有加载这个class,则调用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); 

该方法会调用check()方法来判断这个类是否已经初始化,并且通过checkName(name)来判断由name指定的这个类是否存在***调用findBootstrapClass(name):

Java代码

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

而这个findBootstrapClass方法是一个native方法,这是我们的root loader,这个载入方法并非是由JAVA所写,而是C++写的,它会最终调用JVM中的原生findBootstrapClass方法来完成类的加载。

如果上面两个都找不到,则使用findClass(name)来查找指定类名的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中的说明:

使用指定的二进制名称查找类。此方法应该被类加载器的实现重写,该实现按照委托模型来加载类。在通过父类加载器检查所请求的类后,此方法将被 loadClass 方法调用。默认实现抛出一个ClassNotFoundException。

所以,我们在自定义类中,只需要重写findClass()即可。

MyClassLoader类:

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

该类中通过调用defineClass(String name, byte[] b, int off, int len)方法来定义一个类:

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加载类时有一个局限,必需指定.class文件,而不能指定.jar文件。该类中的大部分代码是从网上搜索到的,是出自一牛人之笔,只是不知道原帖在哪,希望不会被隐藏。

MainClassLoader类:

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

***是一个简单的Test测试类:

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内存泄露的理解与解决
  2. 精解Java中代理模式的实现
  3. Java自带的Future多线程模式
  4. 解析Java finally的神秘面纱
  5. 调用Java NIO提高文件读写速度

 

责任编辑:林师授 来源: huangcanqin的博客
相关推荐

2023-10-19 09:14:34

Java开发

2011-03-16 09:26:41

ReadWriteLoJava

2010-10-13 10:24:38

垃圾回收机制JVMJava

2017-08-17 15:13:52

PostgreSQL MVCC机制

2009-06-17 09:40:01

JBoss的class

2010-09-17 13:02:11

JAVA反射机制

2015-10-26 09:25:42

2021-10-24 06:50:52

AndroidClassLoaderJava

2023-10-20 09:51:00

编程开发

2010-04-26 10:44:27

Oracle SCN

2010-01-25 18:24:11

C++

2011-12-12 10:19:00

JavaNIO

2011-12-12 10:33:47

JavaNIO

2011-07-01 15:04:49

Qt 内省

2009-07-08 14:06:22

ClassLoaderJDK源码

2023-11-08 14:21:51

Python拷贝

2010-10-08 10:42:30

2020-11-06 00:50:16

JavaClassLoaderJVM

2010-08-13 14:19:44

Flex绑定机制

2011-04-07 17:27:52

Policing
点赞
收藏

51CTO技术栈公众号