与普通程序不同的是,Java程序(class文件)并不是本地的可执行程序。当运行Java程序时,首先运行JVM(Java虚拟机),然后再把Java class加载到JVM里头运行,负责加载Java class的这部分就叫做Class Loader。中文叫做类加载器。
Java虚拟机中可以安装多个类加载器,系统默认三个主要类加载器,每个类负责加载特定位置的类:BootStrap,ExtClassLoader,AppClassLoader
类加载器也是Java类,因为其他是java类的类加载器本身也要被类加载器加载,显然必须有第一个类加载器不是不是java类,这正是BootStrap。
Java虚拟机中的所有类装载器采用具有父子关系的树形结构进行组织,在实例化每个类装载器对象时,需要为其指定一个父级类装载器对象或者默认采用系统类装载器为其父级类加载。
一般类加载器的加载过程如下:
调用 findLoadedClass() 来查看是否存在已装入的类。
2>如果没有,那么获取class文件的原始字节。(通过IO从文件系统,来自网络的字节流等)
如果已有原始字节,调用 defineClass() 将它们转换成 Class 对象。
如果没有原始字节,然后调用 findSystemClass() 查看是否从本地文件系统获取类
如果 resolve 参数是 true,那么调用 resolveClass() 解析 Class 对象。
如果还没有类,返回 ClassNotFoundException。
否则,将类返回给调用程序。
类加载器之间的父子关系和管辖范围图
类加载器的委托机制
当Java虚拟机要加载一个类时,到底派出哪个类加载器去加载呢?
Ø首先当前线程的类加载器去加载线程中的第一个类。
Ø如果类A中引用了类B,Java虚拟机将使用加载类A的类装载器来加载类B。
Ø还可以直接调用ClassLoader.loadClass()方法来指定某个类加载器去加载某个类。
每个类加载器加载类时,又先委托给其上级类加载器。
Ø当所有祖宗类加载器没有加载到类,回到发起者类加载器,还加载不了,则抛ClassNotFoundException,不是再去找发起者类加载器的儿子,因为没有getChild方法,即使有,那有多个儿子,找哪一个呢?
Ø对着类加载器的层次结构图和委托加载原理,解释先前将ClassLoaderTest输出成jre/lib/ext目录下的itcast.jar包中后,运行结果为ExtClassLoader的原因。
每个ClassLoader本身只能分别加载特定位置和目录中的类,但它们可以委托其他的类装载器去加载类,这就是类加载器的委托模式。类装载器一级级委托到BootStrap类加载器,当BootStrap无法加载当前所要加载的类时,然后才一级级回退到子孙类装载器去进行真正的加载。当回退到最初的类装载器时,如果它自己也不能完成类的装载,那就应报告ClassNotFoundException异常。
l有一道面试,能不能自己写个类叫java.lang.System,为了不让我们写System类,类加载采用委托机制,这样可以保证爸爸们优先,也就是总是使用爸爸们能找到的类,这样总是使用java系统提供的System。
把先前编写的类加入到jdk的rt.jar中,会有怎样的效果呢?不行!!!看来是不能随意将自己的class文件加入进rt.jar文件中的。
- public class Test {
- public static void main(String[] args) {
- //加载器:sun.misc.Launcher$AppClassLoader
- System.out.println(Test.class.getClassLoader().getClass().getName());
- //加载器:BootStrap(loader为null的情况)
- System.out.println(System.class.getClassLoader());//
- System.out.println("----------------查看类加载器的层次结构关系-------------------");
- ClassLoader loader = Test.class.getClassLoader();
- while(loader != null){
- System.out.println(loader.getClass().getName());
- loader = loader.getParent();
- }
- System.out.println(loader);
- }
- /**
- * 运行结果:
- * sun.misc.Launcher$AppClassLoader
- null
- ----------------查看类加载器的层次结构关系-------------------
- sun.misc.Launcher$AppClassLoader
- sun.misc.Launcher$ExtClassLoader
- null
- */
- }