类加载器顾名思义就是加载类的一种工具,Java虚拟机中可以安装多个类加载器,系统默认三个主要类加载器,每个类负责加载特定位置的类:BootStrap,ExtClassLoader,AppClassLoader类加载器也是Java类,因为其他是java类的类加载器本身也要被类加载器加载,显然必须有第一个类加载器不是不是java类,这正是BootStrap。Java虚拟机中的所有类装载器采用具有父子关系的树形结构进行组织,在实例化每个类装载器对象时,需要为其指定一个父级类装载器对象或者默认采用系统类装载器为其父级类加载。
每个类加载器都会有自己的管辖范围,BootStrap专门加载jre包里面的类,ExtClassLoader专门加载ext目录下的jre里面的类,AppClassLoader专门记在classpath指定的所有jar或目录。
代码来说明放置在不同位置的类确实由不同的类加载器加载的:
System.out.println(ClassLoaderTest.class.getClassLoader().getClass().getName());
//将上面语句的测试类改为System则抛NullPointerException,这两个类存放位置不同
System.out.println(System.class.getClassLoader().getClass().getName());
改为System.out.println(System.class.getClassLoader());打印的结果为null。
代码实现层次结构关系:
ClassLoader loader =ClassLoaderTest.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
三:类加载器的委托机制
每个ClassLoader本身只能分别加载特定位置和目录中的类,但它们可以委托其他的类装载器去加载类,这就是类加载器的委托模式。类装载器一级级委托到BootStrap类加载器,当BootStrap无法加载当前所要加载的类时,然后才一级级回退到子孙类装载器去进行真正的加载。当回退到最初的类装载器时,如果它自己也不能完成类的装载,那就应报告ClassNotFoundException异常。
四:自定义类加载器的编写原理
自定义的类加载器必须继承ClassLoader,loadClass(String name)方法与findClass方法,defineClass方法自定义的加载类使用的时候先调用父类加载器,如果父类加载器管辖的范围内没有找到要加载的类文件,则会执行findClass方法,findClass方法返回一个Class对象,返回的Class对象可以用defineClass方法,将某个类的字节码文件读取流写入到ByteArray输出流中,将字节码对象转换成Byte数组。这样可以写自己的加密类
五:类加载器的高级问题实验分析:
JVM在编译某个类A的时候,会采用类加载的委托机制,调用父类加载器,如果父类加载器中有类A的class文件,则使用父类加载器加载,如果A中使用了其他类B,如果父类加载器的加载目录下找不到类B,则会出现错误。