类加载子系统包括3个阶段:加载阶段、链接阶段、初始化阶段。
类加载子系统阶段1:加载阶段
对于下图的加载步骤1:个人理解是由于每个class文件的文件名和类名是一致的,类加载器就通过class文件名获取到了这个类的全限定名,然后将每一个class文件转换成二进制字节流。
这里4者是包含关系,不是继承关系的原因是:引导类加载器本身是C/C++语言编写,无法与其它类加载器继承或者被继承。另外,这些类加载器的功能也各不相同。
其实上图中说加载扩展类和应用程序类加载器,并制定为他们的父类加载器是不准确的。因为前面刚说过,引导类加载器并不是其它类加载器的父类,其它的类加载器也需要它来进行加载。
注意:==当我们试图获取比如rt.jar包中的类的类加载器时,获取到的类加载器为null。==代码如下所示:
package com.atguigu.java1;
/**
* @author Devin Chen
* @date 2021年1月27日 17:21
*/
public class StringTest {
public static void main(String[] args) {
String str = new String();
//快速生成返回值快捷键:ctrl + alt + v
//获取String类的类加载器
ClassLoader classLoader = str.getClass().getClassLoader();
//获取到String类的类加载器为null,是因为String类的类加载器为引导类加载器,而引导类加载器是C/C++编写的,是获取不到的
System.out.println("String类的类加载器为:" + classLoader);
// StringTest test = new StringTest();
// System.out.println("自定义的StringTest的类加载器为:" + test.getClass().getClassLoader());
}
}
结果:
将最后2行取消注释,可获得自定义类的类加载器。
结果:
获取类加载器举例代码:
package com.atguigu.java1;
/**
* @author Devin Chen
* @date 2021年1月27日 17
*/
public class ClassLoaderTest {
public static void main(String[] args) {
//获取系统类加载器(应用程序类加载器)
ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
System.out.println("系统类加载器:" + systemClassLoader);
//获取扩展类加载器
ClassLoader extClassLoader = systemClassLoader.getParent();
System.out.println("扩展类加载器:" + extClassLoader);
//获取引导类加载器
ClassLoader bootstrapClassLoader = extClassLoader.getParent();
System.out.println("引导类加载器:" + bootstrapClassLoader);
//对于用户自定义类来说,默认使用系统类加载器进行加载
ClassLoader classLoader = ClassLoaderTest.class.getClassLoader();
System.out.println("用户自定义类的类加载器:" + classLoader);
//Java核心类库中的类的类加载器
ClassLoader classLoader1 = String.class.getClassLoader();
System.out.println("java核心类库中类的类加载器:" + classLoader1);
}
}
结果:
获取不到引导类加载器的原因,想必大家都已经很清楚了。补充一句:对于用户自定义类来说,默认使用系统类加载器进行加载。
自定义java.lang.String,代码如下:
package java.lang;
public class String {
static {
System.out.println("我是自定义的String类的静态代码块!!!");
}
public static void main(String[] args) {
String s = new String();
}
}
运行结果: