在运行期,一个Java类是由该类的完全限定名(binary name,二进制名)和用于加载该类的定义类加载器(defining loader)所共同决定的。如果同样名字(即相同的完全限定名)的类是由两个不同的加载器所加载,那么这些类就是不同的,即便.class文件的字节码完全一样,并且从相同的位置加载亦是如此。
编写MyTest23
public class MyTest23 {
public static void main(String[] args) {
// 查看加载器位置
System.out.println(System.getProperty("sun.boot.class.path"));
System.out.println(System.getProperty("java.ext.dirs"));
System.out.println(System.getProperty("java.class.path"));
}
}
java -Dsun.boot.class.path=./ com.yshuoo.jvm.classloader.MyTest23
执行上面的命令
虚拟机初始化出错,在Oracle的Hotspot实现中,系统属性sun.boot.class.path如果修改错了,则运行出错。因为Object类根据类的初始化顺序一定要初始化,在当前./里没有,所以会出错。
扩展类加载器与系统类加载器由谁加载的?
内建于JVM中的启动类加载器会加载java.lang.ClassLoader以及其他的Java平台类,当JVM启动时,一块特殊的机器码会运行,它会加载扩展类加载器与系统类加载器,这块特殊的机器码叫做启动类加载器(Bootstrap)
启动类加载器并不是Java类,而其他的加载器则都是Java类,启动类加载器是特定于平台的机器指令,它负责开启整个加载过程。所有的类加载器(除了启动类加载器)都被实现为Java类。不过总归要有一个组件来加载第一个Java类加载器,从而让整个加载过程能够顺利进行下去,加载第一个纯Java类加载器就是启动类加载器的职责。
启动类加载器还会负责加载供JRE正常运行所需要的基本组件,这包括java.util与java.lang包中的类等等。
System.out.println(ClassLoader.class.getClassLoader());
执行结果是null,可以证明ClassLoader是由启动类加载器加载的。
/**
* 扩展类加载器与系统类加载器是Launcher的私有静态方法
*/
System.out.println(Launcher.class.getClassLoader());
扩展类加载器与系统类加载器也是由启动类加载器所加载的
/**
* 说明默认情况下系统会指向AppClassLoader
*/
System.out.println(System.getProperty("java.system.class.loader"));
将系统类加载器设置成我们自己的类加载器
java -Djava.system.class.loader=com.yshuoo.jvm.classloader.MyTest16 com.yshuoo.jvm.classloader.MyTest23
public MyTest16(ClassLoader parent){
super(parent);
}
根据javadoc在MyTest16里增加这样的构造方法,在控制台里执行
System.out.println(MyTest23.class.getClassLoader());
System.out.println(MyTest16.class.getClassLoader());
System.out.println(ClassLoader.getSystemClassLoader());
微信公众号:二虎程序
源码地址:https://github.com/TigerTurbo/jvm_lecture/blob/master/src/main/java/com/yshuoo/jvm/classloader/