1、类的加载过程
类的加载过程分为三大阶段,分别是加载阶段、连接阶段和初始化阶段,连接阶段又可分为验证、准备、解析三个阶段,如下图所示。
每个阶段做了什么
- 加载阶段:查找并加载类的二进制数据文件,也就是.将class文件中的二进制数据读取到内存中。最终产物是堆内存中的class对象。
- 连接阶段
- 验证:确保类文件符合当前JVM的规范,如class的版本。
- 准备:为类的静态变量分配内存,并为其初始化默认值(不同于初始化阶段)。
- 解析:把类中的符号引用转换为直接引用。
- 初始化阶段:为类的静态变量赋予正确的初始值(代码编写阶段给定的值)。
什么情况下会引发类的加载
- 通过new关键字(注意:构造某个类的数组时也有new关键字,但是此时类并不会初始化)
- 访问类的静态变量、静态方法(注意:引用类的静态常量不会导致类的初始化)
- 对类进行反射操作
- 初始化子类会导致父类的初始化
- 启动类:也就是执行main函数所在的类会导致该类的初始化
2、JVM类别加载器
加载器之间的父子关系
(1)JVM内置三大类加载器
- 根类加载器(BootStrap Classloader)
该加载器是最为顶层的加载器,其没有任何父类加载器,主要负责虚拟机核心类库的加载。它是由C++编写的。 - 扩展类加载器(ExtClassLoader)
扩展类加载器的父类加载器是根加载器,它主要用于加载JAVA_HOME下的jre\lib\ext子目录里的类库。它是由java语言实现的。 - 应用程序类加载器(ApplicationClassLoader)
应用程序类加载器的父类是扩展类加载器,负责加载classpath下的类库资源
(2)自定义类加载器
(3)双亲委派机制
其工作原理的是,如果一个类加载器收到了类加载请求,它并不会自己先去加载,而是把这个请求委托给父类的加载器去执行,如果父类加载器还存在其父类加载器,则进一步向上委托,依次递归,请求最终将到达顶层的启动类加载器,如果父类加载器可以完成类加载任务,就成功返回,倘若父类加载器无法完成此加载任务,子加载器才会尝试自己去加载,这就是双亲委派模式,
作用:避免了类的重复加载
3、线程上下文加载器
为什么需要线程上下文类加载器