/*类加载器*/ 把.class文件从硬盘上加载出来,将类的字节码(二进制)加载到内存中
/*类加载器及其委托机制*/
Java虚拟机中可以安装多个类加载器(可以自己编写),系统默认三个主要类加载器,每个类负责加载特定位置的类 BootStrap,ExtClassLoader,AppClassLoader类加载器也是Java类,因为其他是java类的类加载器本身也要被类加载器加载,显然必须有第一个类加载器不是java类,这正是BootStrap(C++)
Java虚拟机中的所有类装载器采用具有父子关系的树状结构进行组织,在实例化每个类装载器对象时,需要为其制定一个父级类装载器对象 或者默认采用系统类装载器为其父级类加载 /*类加载器之间的父子关系和管辖范围*/ BootStrap ---- JRE/lib/rt.jr (一般java中自己有的类:如System类,Map等) | ExtClassLoader ---- JRE/lib/ext/* (第三方的类) */ | AppClassLoader ----- CLASSPATH指定的所有jar或目录 (一般自己编写的类) | MyClassLoader(自己编写的ClassLoader) /*委托机制*/ 1.首先当前线程的类加载器去加载线程中的第一个类, 2.如果类A中引用了类B,Java虚拟机将使用加载类A的类装载器来加载B 3.还可以直接调用ClassLoader.loaderClass()方法来指定某个类加载器去加载某个类 /*每个类加载器加载类时,又先委托给其上级类加载器*/ 1.当所有祖宗类加载器没有加载到类,回到发起者类加载器,还加载不到,则抛出ClassNotFoundException, 不是再去找发起者类加载器的儿子,因为没有getChild()方法,即使有,那有多个儿子,找哪一个呢? 如:System类,AppClassLoader先发出申请,然后委托给ExtClassLoader,ExtCLassLoader同样将其抛出,委托给BootStrap , 然后BootStrap 查找时候有这个类,然后加载这个类,如果BootStrap 没有找到这个类,则向下一级ExtClassLoader,以此类推,直到发起者 /*类加载器的加载都要让BootStrap 这个祖宗先查一遍,这样方便管理*//*编写自己的类加载器*/1.自定义的类加载器必须继承ClassLoader2.loadClass方法(调用)与findClass方法(覆写)3.defineClass方法父类 -->loadClass/findClass/得到class文件的转换成字节码-->defineClass()
只需要覆盖findClass方法,将某个类变成二进制数据,然后调用defineClass,将二进制数据变成相应的字节码文件loadClass方法不要覆盖,保留类加载器的委托机制/*loadClass() 方法里面分为两部分*/ /*第一部分*/是主要流程:即把一个类加载进内存的时候它要先去一级一级地找父类,看父类能不能加载,如果父类能加载的话, 就不用再返回到了,让该类加载, /*这部分就是委托机制*/ /*第二部分*/是细节部分:就是如果由该类自己加载该怎么加载。也就是findClass()部分 public abstract class ClassLoader { private loadClass() { //loadClass()中的主要流程 } findClass(); //loadClass() 中的部分细节。 } public abstract findClass() { } 所以我们自定义加载器只需复写ClassLoader类中的findClass()方法就行了~~ (loadClass方法如果在前面的父类都无法加载,就会调用findClass加载) MyClassLoader.loadClass() //自定义的加载器也是调用loadClass即可主要方向:loadClass() -->loadClass() --> defineClass(String name,byte[] bytes, off, len)
-->findClass(String name)