文章目录
1 概述
java中数据类型分数为基本数据类型和引用数据类型,基本数据类型由虚拟机预先定义,引用数据类型则需要进行类的加载。
2 loading 加载
2.1 加载概述
加载就是将java类的字节码文件加载到机器内存中,在内存中构建java类的原型——类模板对象。所谓的类模板对象,其实就是java类在jvm内存中的一个块照,jvm将字节码文件中解析出的常量池,类字段,类方法等信息存储到类模板中,jvm在运行期就能通过类模板获取java类中任意信息。(反射就是基于此基础)
2.2 加载流程
查找并加载类的二进制数据,生成class的实例
-
通过类的全名,获取类的二进制数据流
-
解析类的二进制数据流为方法区的数据结构(java类模型)
-
创建java.lang.Class 类实例表示该类型
2.2.1 二进制流的获取方式
只要所读取的字节码符合jvm规范即可
-
虚拟机读取class后缀的文件(常见)
-
读入jar,zip等数据包
-
使用http之类的协议通过网络加载
在获取到类的二进制信息后,jvm处理这些数据,并最终产生java.lang.class的实例
2.2.2 类模型及class类实例的位置
-
类模型位置
加载的类在jvm中创建相应的类结构,类结构存储在方法区(jdk1.8之前:永久代,jdk1.8之后 元空间) -
class实例位置
类将class文件加载到元空间后,会在堆中创建一个java.lang.class对象,用于封装方法区的数据结构,此过程在类加载过程中创建。
3 linking 连接阶段
3.1 验证
验证阶段包括:格式验证,语义检查,字节码验证,以及符号引用验证
-
其中格式验证会和加载阶段一起执行
-
格式验证之外的验证操作会在方法区中进行
3.2 准备阶段
为类的静态变量分配内存,为其初始化为默认值
注意:这里不包括基本数据类型字段用static final修饰的情况,因为final在编译的时候就会分配了,准备阶段会显式赋值。
3.3 解析阶段
解析阶段是将类,接口,字段和方法的符号引用转为直接引用。
解析操作往往会在jvm执行完初始化之后在执行。
如果使用符号引用,虚拟机其实也不知道具体引用的类的内存地址,那么也就无法真正的调用到该类,所以要把符号引用转为直接引用,这样就能够真正定位到类在内存中的地址,如果符号引用转直接引用失败,就说明类还没有被加载到内存中,就会报错
4 初始化阶段
为类的静态变量赋予正确的初始值
4.1 流程
到了初始化阶段,此时类才会执行Java字节码(只有到了初始化,才会执行类中定义的java程序)重要的工作时执行类的初始化方法:()方法
-
由静态成员的赋值以及static语句块合并产生
-
该方法仅有java编译器生成,并由jvm进行调用。
注意:在加载一个类之前,虚拟机会试图加载该类的父类,因此父类的总是在子类之前被调用,也就是说父类的static块优先于子类
结论: 在连接阶段的准备环节赋值情况:
1.对于基本数据类型的字段,static final 修饰(直接赋值常量,而不是方法调用) 则是在连接阶段
2.对于string来说,如果是字面量方式赋值+static final修饰,则也是在连接阶段
3.除了上述的以为的都在初始化阶段赋值
5 类的主动使用与被动使用
-
主动使用(类会被初始化执行)
-
被动使用(不会执行初始化,但会被加载)
class只有在必须首次的使用时才会被装载
5.1 主动使用
-
创建一个类的实例,比如:new,反射,序列化
-
调用类的静态方法
-
使用类,接口的静态字段时
-
使用 java.lang.reflect 包中的方法反射类的方法时
-
初始化子类时,若发现父类未初始化,则先进行父类初始化
注意:不会初始化他所实现的接口,当程序首次使用接口的特定字段时才会初始化。 -
如果接口定义了default方法,则接口在会被在所实现类之间进行初始化
-
虚拟机启动时,用户指定要执行的主类(含main方法),虚拟机会初始化这个类
-
当调用methodHandle实例时,初始化该methodHandle指向的方法所在的类
5.2 被动使用
被动使用不会引起类的初始化。并不是在代码中出现的类,就一定会被 加载或者初始化。如果不符合主动使用的条件,类就不会初始化。
-
当访问一个静态字段时,只有真正声明这个字段的类才会被初始化。
- 当通过子类引用父类的静态变量时,不会导致子类初始化
-
通过数组定义类引用
-
引用常量不会触发此类或接口的初始化
-
调用classloader类的loadclass方法加载一个类
注意:没有初始化的一个类不意味着没有被加载
6 类的使用
类被加载完后就可以使用类的属性 或者创建对象了
7类的卸载
7.1 类,类加载器,类实例之间关系
通过类加载器可以加载class对象,另一方面 一个class对象总是会引用他的类加载器,调用class对象的getClassloader()方法,会获得它的类加载器。
一个类的实例总是引用代表这个类的class对象 ,在object中定义了getclass()方法,代表所属类class对象的引用,此外java类都有静态属性class,它引用代表这个类的class对象。
7.2 类的生命周期
当sqmple类的class对象不再被引用时,即不可触及时,class对象就会结束生命周期,sample类在方法区内数据也会被卸载
一个类何时结束生命周期,取决于代表它的class对象何时结束生名周期