在路上

 找回密码
 立即注册
在路上 站点首页 学习 查看内容

Java类加载机制(二)

2017-2-7 13:39| 发布者: zhangjf| 查看: 544| 评论: 0

摘要: 类加载器原理将class文件字节码内容加载到内存中,并将这些静态数据转换为方法区的运行时数据结构,在堆中生成一个代表这个类的java.lang.Class对象,作为方法区类数据访问的入口复制代码 类加载器树状结构、双亲委 ...
类加载器原理
  1. 将class文件字节码内容加载到内存中,并将这些静态数据转换为方法区的运行时数据结构,在堆中生成一个代表这个类的java.lang.Class对象,作为方法区类数据访问的入口
复制代码
类加载器树状结构、双亲委托机制 类加载器树状结构

引导类加载器 用来加载Java的核心库(JAVA_HOME/jre/lib/rt/jar,或sun.boot.class.path路径 下的内容),是用原生的代码(c++)实现的,并不继承java.lang.ClassLoader。 加载扩展类加载器和应用程序类加载器。并指定它们的父类加载器。

扩展类记载器 用来加载Java的扩展库(JAVA_HOME/jre/ext/*.jar,或java.ext.dirs路径下的内容)。 Java虚拟机的实现会提供一个扩展库目录。该类加载器在此目录里面查找并加载Java类。

应用程序类加载器 它根据Java应用的类路径(classpath,java.class.path路类) 一般来说,Java应用的类都是由它来完成加载的。由sun.misc.Launcher$AppClassLoader实现。

自定义类加载器 开发人员可以通过继承java.lang.ClassLoader类的方式实现自己的类加载器,以满足一些特殊的需要。

类加载器图示

  1. public class Demo {
  2. public static void main(String[] args) {
  3. //获取应用程序类加载器
  4. System.out.println(ClassLoader.getSystemClassLoader());
  5. //获取扩展类加载器
  6. System.out.println(ClassLoader.getSystemClassLoader().getParent());
  7. //获取引导类加载器
  8. System.out.println(ClassLoader.getSystemClassLoader().getParent().getParent());
  9. //获取classpath
  10. System.out.println(System.getProperty("java.class.path"));
  11. }
  12. }
复制代码
双亲委托机制

代理模式:交给其他类加载器加载指定的类

双亲委托机制

(1)就是当某个特定的类加载器接到加载类的请求的时候,首先委托给其父类(父类如果有父类一直向上追溯),直到父类加载器无法加载时,该加载器进行加载。

(2)双亲委托机制是为了保证Java核心库的类型安全。

这种机制保证不会加载到用户自定义的java.lang.Class类的情况

(3)类加载器除了用于加载类,也是安全最基本的屏障。

自定义类加载器
  1. 如何实现自定义类加载器:
  2. (1)继承java.lang.ClassLoader
  3. (2)检查所请求的类型是否已经被这个类加载器加载到命名空间,如果已经被加载直接返回。
  4. (3)委派给父类加载(也可以不委派,这个程序控制)。
  5. (4)调用自定义加载器findClass()方法获取字节码,然后调用defineClass()导入类型到方法区。
复制代码
  1. public class FileSystemClassLoader extends ClassLoader {
  2. String rootDir;
  3. public FileSystemClassLoader(String rootDir) {
  4. this.rootDir = rootDir;
  5. }
  6. @Override
  7. protected Class<?> findClass(String name) throws ClassNotFoundException {
  8. Class c = findLoadedClass(name);
  9. if (c != null) {
  10. return c;
  11. } else {
  12. ClassLoader parent = this.getParent();
  13. try {
  14. c = parent.loadClass(name);
  15. }catch (Exception e){
  16. e.printStackTrace();
  17. }
  18. if (c != null) {
  19. return c;
  20. } else {
  21. byte[] classData = getClassData(name);
  22. if (classData == null) {
  23. throw new ClassNotFoundException("自定义类加载器没有加载到");
  24. } else {
  25. c = defineClass(name, classData, 0, classData.length);
  26. }
  27. }
  28. }
  29. return c;
  30. }
  31. private byte[] getClassData(String className) {
  32. String path = rootDir + "/" + className.replace(".", "/") + ".class";
  33. InputStream is = null;
  34. ByteArrayOutputStream baos = new ByteArrayOutputStream();
  35. try {
  36. is = new FileInputStream(path);
  37. byte[] buffer = new byte[1024];
  38. int temp = 0;
  39. while ((temp = is.read(buffer)) != -1) {
  40. baos.write(buffer,0,temp);
  41. }
  42. return baos.toByteArray();
  43. } catch (Exception e) {
  44. return null;
  45. } finally {
  46. if(is != null){
  47. try {
  48. is.close();
  49. } catch (IOException e) {
  50. e.printStackTrace();
  51. }
  52. }
  53. if(baos != null){
  54. try {
  55. is.close();
  56. } catch (IOException e) {
  57. e.printStackTrace();
  58. }
  59. }
  60. }
  61. }
  62. }
  63. public class TestMyClassLoader {
  64. public static void main(String[] args) throws ClassNotFoundException {
  65. FileSystemClassLoader loader = new FileSystemClassLoader("/Users/wjk/Desktop");
  66. FileSystemClassLoader loader1 = new FileSystemClassLoader("/Users/wjk/Desktop");
  67. Class c = loader.loadClass("com.Hello");
  68. Class c1 = loader1.loadClass("com.Hello");
  69. Class c2 = loader.loadClass("com.Hello");
  70. Class c3 = loader.loadClass("java.lang.String");
  71. System.out.println(c.hashCode());//被两个类加载器加载的同一个类,JVM认为是不同的(c和c1的hashCode值不一样)
  72. System.out.println(c1.hashCode());
  73. System.out.println(c2.hashCode());
  74. System.out.println(c.getClassLoader());//使用的是自定义的类加载器
  75. System.out.println(c3.getClassLoader());//使用的是引导类加载器
  76. }
  77. }
  78. //结果
  79. 1725154839
  80. 1670675563
  81. 1725154839
  82. classLoaderTest.FileSystemClassLoader@5e2de80c
  83. null
复制代码
线程上下文类加载器 服务器类加载器原理和OSGI介绍

来自: http://my.oschina.net/u/2361475/blog/603798

最新评论

小黑屋|在路上 ( 蜀ICP备15035742号-1 

;

GMT+8, 2025-7-9 02:20

Copyright 2015-2025 djqfx

返回顶部