------- android培训、java培训、期待与您交流! ----------
1. 基本类加载器
(1)BootStrap
rt.jar(RunTime运行库),包括java基本类库
(2)ExtClassLoader
ext目录下的jar包
(3)AppClassLoader
加载classpath指向的jar和目录
2.类加载器间的关系
类加载器也是java类,因为其他是java类的类加载器本身也要被类加载器加载,显然必须有第一个类加载器不是java类,这正是BootStrap。
虚拟机中的所有类加载器采用具有父子关系的树形结构进行组织,,子实例化每个类加载器对象时,需要为其指定一个父级类加载器对象或者默认采用系统类加载器为其父级类加载器。
类加载器之间的父子关系:
BootStrap<--------ExtClassLoader<--------AppClassLoader<--------MyClassLoader
自定义类加载器为AppClassLoader的子类。
3.类加载器的委托机制
当java虚拟机要加载一个类时,到底要派出哪个类加载器去加载?
首先,当前线程的类加载器去加载线程中的第一个类,如果类A中引用了类B,java虚拟机将使用加载类A的加载器来加载类B,还可以直接调用classLoader.loadCass()方法来指定某个类加载器去加载某个类。
每个类加载器加载类时,先会委托给上级类加载器:
当所有祖宗类加载器没有加载到类,回到发起者类加载器;若还加载不了,则抛出ClassNotFoundException,而不是再去找发起者类加载器的儿子,因为没有getChild()方法,即使有,也不能确定找哪一个。
4.自定义类加载器
自定义类加载器必须继承ClassLoader,调用loader加载类时,会调用findClass,寻找类文件,findClass调用defineClass函数,将class文件转化成内存中的字节码文件。
重写loadClass很困难,因为它会改变所有类加载器的loadClass方法,所有重写findClass,指定需要加载的目录,还要在findClass中调用defineClass返回类文件的可执行字节码文件。