jvm底层-类加载与oop-klass模型


以下是针对jdk1.8

1.1 oop-klass模型

oop 对象
klass 类的元信息

看下klass模型类的继承结构
在这里插入图片描述
从继承关系上也能看出来,类的元信息是存储在原空间(MetespaceObj)。

1.1.1 InstanceKlass

InstanceKlass:普通的Java类在JVM中对应的是instanceKlass类的实例。(形象理解)类加载器将.class文件加载进jvm,将.class文件解析,把类的元信息存储在JVM

InstanceKlass子类三个:

  1. InstanceMirrorKlass(镜像类):用于表示java.lang.Class,Java代码中获取到的Class对象,实际上就是这个C++类的实例,存储在堆区
  2. InstanceRefKlass:用于表示java/lang/ref/Reference类的子类,即强软弱虚4种引用
  3. InstanceClassLoaderKlass:用于遍历某个加载器加载的类

1.1.2 ArrayKlass

ArrayKlass:存储数组类的元信息。Java数组的元信息用ArrayKlass的子类来表示

  1. TypeArrayKlass: 表示基本类型的数组在JVM中的存在形式。
  2. ObjArrayKlass:表示引用类型的数组在JVM中的存在形式。

1.2 类加载的过程

在这里插入图片描述

1.2.1 加载

1、通过类的全限定名(包名+类名)获取存储该类的class文件
2、解析成运行时数据,即instanceKlass实例,存放在方法区
3、在堆区生成该类的Class对象,即instanceMirrorKlass实例

1.2.2 验证

1、文件格式验证
2、元数据验证
3、字节码验证
4、符号引用验证

1.2.3 准备

为静态变量分配内存、赋初值
实例变量是在创建对象的时候完成赋值的,没有赋初值一说

在这里插入图片描述
如果被final修饰,在编译的时候会给属性添加ConstantValue属性,准备阶段直接完成赋值,即没有赋初值这一步

1.2.4 解析

将常量池中的符号引用转为直接引用

简单说就是把间接引用转直接引用

间接引用:指向运行时常量池的引用
直接引用:内存地址

1.2.5 初始化

执行静态代码块,完成静态变量的赋值

细节:

  1. 静态字段、静态代码段,字节码层面会生成clinit方法
  2. 方法中语句的先后顺序与代码的编写顺序相关

1.3 类什么时候加载

JVM加载类是懒加载

  1. new、getstatic、putstatic、invokestatic
  2. 反射
  3. 初始化一个类的子类会去加载其父类
  4. 启动类(main函数所在类)

猜你喜欢

转载自blog.csdn.net/qq_33873431/article/details/112851125