加载(Loading)
- 读取class文件获取二进制流
- 将类的静态存储结构转换为方法区的运行时数据结构(1.7及之前叫永久代,之后叫元空间)
- 在内存中生成一个该类的java.lang.Class对象
链接(Linking)
验证(Verification)
- 文件格式验证、元数据验证、字节码验证、符号引用验证
- 十六进制文件头一定是
CA FE BA BE
准备(Preparation)
- 为类的成员变量在堆空间中分配内存并设置零值(boolen为false,int为0等等)
- 被final修饰的static变量为常量,在编译阶段就分配值了,在准备阶段会显式初始化(即不需要零值初始化)
- 这里不会为实例变量初始化(此时还没有对象),类的成员变量位于方法区,而对象的变量位于堆空间
解析(Resolution)
- 将常量池内的符号引用转换为直接引用,直接引用就是直接指向目标的指针、相对偏移量或一个间接定位到目标的句柄
- 解析动作主要针对类或接口、字段、类方法、接口方法、方法类型等
初始化(Initialization)
- 初始化过程是执行类构造器方法
<clinit>()
的过程,此方法是javac
命令执行时自动收集类中所有类成员变量的赋值操作以及static静态代码块中的语句合并而来
- 构造器方法中的指令按照语句在源文件中出现的顺序执行(此时可以打破先定义后使用的规则,但是此时不可以调用它)
- 若当前类有父类则会先保证父类的
<clinit>()
执行完毕 - 虚拟机会保证一个类的
<clinit>()
在多线程下会被同步加锁
20210506 UPDATE
Java对类的使用方式分为主动使用
和被动使用
,主动使用包含以下七种情况:
- 创建类的实例
- 访问类的静态变量,或者对静态变量赋值
- 调用类的静态方法
- 反射(
Class.forName("com.xxx.xxx")
) - 初始化一个类的子类
- Java虚拟机启动时被标明为启动类的类
- JDK 7 开始提供的动态语言支持(?)
除了以上的情况,其他使用方式均会被视为类的被动使用,被动使用不会导致类的初始化,即不会执行 clinit 方法