一、类加载机制
分为三步:加载,连接(验证,准备,解析),初始化,使用,卸载
加载:
- 将class文件读入内存,并创建一个java.lang.Class对象,程序中使用任何类时,系统都会自动创建一个java.lang.Class对象,系统中所有的类都是java.lang.Class对象的实例。
- 类的加载由类加载器完成的,JVM提供的类加载器叫做系统类加载器,还可以通过继承ClassLoader基类来自定义类加载器
- 通常可以用以下几种方法加载类的二进制数据:
- 从本地系统加载class文件
- 从jar包中加载class文件,如jar包的数据库驱动类
- 通过网络加载class文件
- 把一个Java源文件动态编译并执行加载
连接:
负责把类的二进制数据合并到JRE中,通过三步:
- 验证:确保加载的类信息符合JVM规范,无安全方面问题
- 准备:为类的静态字段分配内存,并设置初始值
- 解析:将类的二进制数据中的符合引用替换成直接引用
什么是符号引用和直接引用?
符号引用:以一组符号来描述所引用的目标,比如你可以是张三 也可以有工号表示。
直接引用:可以指向目标的指针、相对偏移量或者是一个直接或者间接能定位到目标的句柄。
初始化:
主要对类的静态变量赋予正确的初始值,JVM负责对类进行初始化,主要对类变量进行初始化,对静态字段指定的初始值有两种方式:
- 声明时即指定初始值,例如 static int a = 5
- 使用静态代码块为静态字段指定初始值,static{b = 5}
JVM初始化一个类有以下步骤:
- 假如这个类还没有被加载和连接,则程序先加载并连接该类
- 假如该类的父类还没有被初始化,则先初始化其直接父类
- 假如类中有初始化语句,则系统依次执行这些初始化语句
类初始化的时机:
- 创建类的实例时(new、反射、反序列化)
- 调用某个类的静态方法
- 使用某个类或接口的静态字段或对该字段赋值
- 使用反射来强制创建某个类或接口对应的java.lang.Class对象,如Class.forName(“Person”)
- 初始化某个类的子类时,此时该子类的所有父类都会被初始化
- 直接使用java.exe运行某个主类时
- 类加载器及加载机制
类加载器负责把class文件读入内存,并且为之生成对应的java.lang.Class对象。
类的加载:
指将类的.class文件中的二进制数据读入内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个Java.lang.Class对象,用来封装类在方法区内的数据结构
三个类加载器:
1,Bootstrap ClassLoader:最顶层的类加载器,主要加载核心类库,也就是我们环境变量下面%JRE_HOME%\lib下的rt.jar、resources.jar、charsets.jar和class等
2,Extention ClassLoader:扩展类加载器,负责加载JRE的扩展目录,加载目录%JRE_HOME%\lib\ext目录下的jar包和class文件。还可以加载-D java.ext.dirs选项指定的目录。
3,Appclass Classloader: 也称为SystemAppClass,加载当前应用classpath的所有类
- 类加载机制
JVM类加载机制主要有如下3种:
- 全盘负责:所谓全盘负责就是当一个类加载器加载某一个Class时,该Class所依赖和所引用的其他Class也应该由该加载器加载,除非显式的使用另一类加载器加载
- 双亲委派:当一个类加载器收到类加载的请求时,它会把这个类请求委派给父类加载器去完成,依次递归,因此最终的所有加载请求都被委派到最顶层启动类加载器中。只有父类加载器无法加载该类时,子类才尝试从自己类的路径中加载该类。
- 缓存机制:缓存机制会保证所有加载过的Class都会被缓存,当程序需要某个类时,类加载器会先从缓存中搜索,若搜索不到就会获取该类的二进制数据,并转换成Class对象存入缓冲区中。这就是为什么修改了Class后要重启JVM才能生效的原因。
注:本博客参考https://blog.csdn.net/cnahyz/article/details/82219210
https://baijiahao.baidu.com/s?id=1636309817155065432&wfr=spider&for=pc