我们知道 我们在Java中用到的所有的类都是通过类加载器ClassLoader加载到JVM中的,我们还知道 类加载器 也对应着一个类 ,既然这样那么我们会想 那么ClassLoader类是由谁加载的呢?
其实在Java中有许许多多的 类加载器 我们甚至可以写自己的类加载器 。
其中主要三个类加载器(他们是树形关系)是 :
BootStrap : 在java虚拟机启动的时候会利用这个类加载器来加载 JDK安装目录下的 /JRE/LIB/rt.jar 也就是系统默认导入的一些类 例如System类 ,这个类加载器不是类 。只是作为一个java中 类的起源工具
ExpClassLoader 这个类加载器加载JDK安装目录下的 /JRE/LIB/ext 目录中的类 我们只要把 我们的类打包成JAR包 放在这里即可
AppClassLoader 我们在java程序中 classpath对应的类 都有 这个AppClassLoader导入进来
看下面一段代码 :
package me.test;
/** * BootStrap * 加载 JRE/lib/rt.jar 包中的类 包括我们常用到的类 * * ExtClassLoader * 专门家在 JDK/JRE/libEXT/*.jar 中的类 只要把我们的类放在这里 就会被 这个加载器加载 * * AppClassLoader * 加载ClassPath指定的所有jar和目录 * * **/ public class Test1 { public static void main(String []args) { System.out.println(Test1.class.getClassLoader().getClass().getName() ); //获取主类的类加载器System.out.println(System.class.getClassLoader()); //BootStrap 获取System类的类加载器 因为加载器是 BootStrap所以返回null 以为内他不是一个类
ClassLoader l=Test1.class.getClassLoader() ; //获取Test1的类加载器 while(l!=null) //循环出 ClassLoader树 { System.out.println(l.getClass().getName()); l=l.getParent(); } System.out.println(l); }}
ClassLoader的委托模型
比如说我们在加载一个类的时候 AppClassLoader 他先让 BootStrap来加载类 如果 BootStrap已经加载了 那么就返回 。如果找不到这个类那么 BootStrap就传递给ExtClassLoader 来查找 ,和BootStrap一样 如果找到就加载 如果找不到 久继续传递给 AppClassLoader 来加载 如果 AppClassLoader还找不到的话 那么
AppClassLoader 就会跑出 ClassNotFoundException 异常 。。
我们为什么不利用 AppClassLoader 下级的加载器呢? 因为AppClassLoader下级可能有多个 类加载器 多个类加载器 相互独立 ,如果加载类 那么就会导致内存中出现
多份字节码 ,造成不必要的的内存浪费 。。这就是类加载器的委托模型