转载

【Android面试-Java-V04】Java类加载过程

java.lang.Class

Java虚拟机在识别Java class文件,产生了类型的二进制数据后,Java虚拟机必须把这些二进制数据解析为与实现相关的内部数据结构。 装载的最终产品就是Class实例,它称为Java程序与内部数据结构之间的接口 。要访问关于该类型的信息(存储在内部数据结构中),程序就要调用该类型对应的Class实例的方法。 这样一个过程,就是把一个类型的二进制数据解析为方法区中的内部数据结构,并在堆上建立一个Class对象的过程,这被称为"创建"类型。

验证

确认装载后的类型符合Java语言的语义,并且不会危及虚拟机的完整性。

装载时验证
正式验证阶段
符号引用的验证

准备

当Java虚拟机装载一个类,并执行了一些验证之后,类就可以进入准备阶段。 在准备阶段,Java虚拟机为类变量分配内存,设置默认初始值。但在到到初始化阶段之前,类变量都没有被初始化为真正的初始值。

boolean在内部常常被实现为一个int,会被默认初始化为0。

解析

类型经过连接的前两个阶段--验证和准备--之后,就可以进入第三个阶段--解析。解析的过程就是在类型的常量池总寻找类、接口、字段和方法的符号引用, 把这些符号引用替换为直接引用的过程

  • 类或接口的解析 :判断所要转化成的直接引用是数组类型,还是普通的对象类型的引用,从而进行不同的解析。

  • 字段解析 :对字段进行解析时,会先在本类中查找是否包含有简单名称和字段描述符都与目标相匹配的字段,如果有,则查找结束;如果没有,则会按照继承关系从上往下递归搜索该类所实现的各个接口和它们的父接口,还没有,则按照继承关系从上往下递归搜索其父类,直至查找结束,

初始化

为类变量赋予“正确”的初始值。这里的“正确”的初始值是指程序员希望这个类变量所具备的初始值。 所有的类变量(即静态量)初始化语句和类型的静态初始化器都被Java编译器收集在一起,放到一个特殊的方法中。 对于类来说,这个方法被称作类初始化方法;对于接口来说,它被称为接口初始化方法。在类和接口的class文件中,这个方法被称为 <clinit>

初始化类的步骤:

1. 如果存在直接父类,且直接父类没有被初始化,先初始化直接父类。
2. 如果类存在一个类初始化方法,执行此方法。
复制代码

这个步骤是递归执行的,即第一个初始化的类一定是 Object初始化接口并不需要初始化它的父接口。

Java虚拟机必须确保初始化过程被正确地同步。如果多个线程需要初始化一个类,仅仅允许一个线程来进行初始化,其他线程需等待。

这个特性可以用来写单例模式。

<clinit>() 方法

  • 对于静态变量和静态初始化语句来说:执行的顺序和它们在类或接口中出现的顺序有关。
  • 并非所有的类都需要在它们的 class 文件中拥有 <clinit>() 方法, 如果类没有声明任何类变量,也没有静态初始化语句,那么它就不会有 <clinit>() 方法。如果类声明了类变量,但没有明确的使用类变量初始化语句或者静态代码块来初始化它们,也不会有 <clinit>() 方法。如果类仅包含静态 final 常量的类变量初始化语句,而且这些类变量初始化语句采用编译时常量表达式,类也不会有 <clinit>() 方法。 只有那些需要执行Java代码来赋值的类才会有 <clinit>()
  • final 常量:Java虚拟机在使用它们的任何类的常量池或字节码中直接存放的是它们表示的常量值。

主动使用和被动使用

所有Java虚拟机实现必须在每个类或接口首次主动使用时初始化。

  • 当创建某个类的新实例时(new、反射、克隆、序列化)
  • 调用某个类的静态方法
  • 使用某个类或接口的静态字段,或对该字段赋值(用final修饰的静态字段除外,它被初始化为一个编译时常量表达式)
  • 当调用Java API的某些反射方法时。
  • 初始化某个类的子类时。
  • 当虚拟机启动时被标明为启动类的类。

除以上六种情况,所有其他使用Java类型的方式都是被动的,它们不会导致Java类型的初始化。

对于接口来说,只有在某个接口声明的非常量字段被使用时,该接口才会初始化,而不会因为事先这个接口的子接口或类要初始化而被初始化。

使用一个非常量的静态字段只有当类或接口的确声明了这个字段时才是主动使用。比如:类中声明的字段可能被子类引用;接口中声明的字段可能被子接口或实现了这个接口的类引用。对于子类、子接口或实现了接口的类来说,这是被动使用--不会触发它们的初始化。只有当字段的确是被类或接口声明的时候才是主动使用。

原文  https://juejin.im/post/5e8997176fb9a03c4a496a24
正文到此结束
Loading...