JVM之类加载器子系统

1、类加载器子系统负责从文件系统或者网络中加载Class文件,Class文件在文件开头有特定的文件标识。

2、ClassLoader只负责Class文件的加载,至于他是否可以运行,则是由ExecutionEngine决定。

3、加载的类信息存放于一块称为方法区的内存空间。除了类的信息外,方法区中还会存放运行时常量池的信息,可能还包括字符串字面量和数字常量(这部分常量信息是Class文件中常量池部分的内存映射)。

一、链接

JVM之类加载器子系统

1、加载

(1)通过一个类的全限定名获取此类的二进制字节流

(2)将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构

(3)在内存中生成一个代表这个类的Class对象。

2、验证(Verify)

(1)确保Class文件的字节流中包含信息符合虚拟机的要求,保证被加载类的正确性,不会危害虚拟机本身。

(2)验证方式:文件格式验证,元数据验证,字节码验证,符号引用验证

3、准备(Prepare)

(1)为类变量分配内存并设置该类变量的默认初始值

         注意:这里不包括用final修饰的static变量,因为final在编译时就会分配,准备阶段会显示初始化。

(2)实例变量不分配内存和初始化,类变量会分配在方法区中,而实例变量是会随着对象一起分配到java堆中。

4、解析(Resolve)

(1)将常量池内的符号引用转换为直接引用的过程

(2)事实上,解析操作往往会随着JVM在执行完初始化之后再执行。

(3)符号引用就是一组符号来描述所引用的目标。

          符号引用的字面量形式明确定义在《java虚拟机规范》的Class文件格式中。

           直接引用就是直接指向目标的指针,相对偏移量或者一个间接定位到目标的句柄。

(4)解析动作主要针对类或者接口、字段、类方法、接口方法、方法类型等。

5、初始化

(1)初始化阶段就是执行类构造器方法<clinit>()过程,该方法不需定义,是javac编译器自动收集类中的所有类变量的赋值动作和静态代码块中的语句合并而来。

         换句话说,如果类中没有静态变量或者静态代码块,则不会出现<clinit>()方法

(2)<clinit>()构造器方法中指令按语句在源文件中出现的顺序执行

(3)<clinit>()不同于类的构造器,在虚拟机中类的构造器是<init>()

(4)如果类有父类,JVM会保证子类的<clinit>()执行前,父类的<clinit>()已经执行完毕。

(5)虚拟机必须保证一个类的<client>()方法在多线程下被同步加锁。

 

 

相关文章:

  • 2021-04-26
  • 2021-12-17
  • 2021-11-25
  • 2022-03-08
  • 2021-10-10
  • 2021-07-27
  • 2021-09-16
  • 2021-09-20
猜你喜欢
  • 2022-12-23
  • 2021-07-14
  • 2022-12-23
  • 2021-05-23
  • 2022-12-23
  • 2021-07-31
相关资源
相似解决方案