JVM详解之:类的加载连接和初始化

简介

有了java class文件以后,为了让class文件转换成为JVM能够真正运行的结构,须要经历加载,连接和初始化的过程。java

这三个过程是怎么工做的呢?在本文中你将会找到答案。数组

加载

JVM能够分为三大部分,五大空间和三大引擎,要讲起来也不是特别复杂,先看下面的整体的JVM架构图。架构

从上面的图中,咱们能够看到JVM中有三大部分,分别是类加载系统,运行时数据区域和Execution Engine。jvm

加载就是根据特定名称查找类或者接口的二进制表示,并根据此二进制表示来建立类和接口的过程。spa

运行时常量池

咱们知道JVM中有一个方法区的区域,在JDK8中,方法区的实现叫作元空间。这个元空间是存放在本地内存中的。对象

方法区中存放着每一个class对应的运行时常量池。blog

当类或者接口建立的时候,就会经过class文件中定义的常量池来构建运行时常量池。接口

运行时常量池中有两种类型,分别是symbolic references符号引用和static constants静态常量。内存

其中静态常量不须要后续解析,而符号引用须要进一步进行解析处理。ci

静态常量分为两个部分:String常量和数字常量。

String常量是对String对象的引用,是从class中的CONSTANT_String_info结构体构建的。

数字常量是从class文件中的CONSTANT_Integer_info, CONSTANT_Float_info, CONSTANT_Long_info和 CONSTANT_Double_info 构建的。

符号引用也是从class中的constant_pool中构建的。

对class和interface的符号引用来自于CONSTANT_Class_info。

对class和interface中字段的引用来自于CONSTANT_Fieldref_info。

class中方法的引用来自于CONSTANT_Methodref_info。

interface中方法的引用来自于CONSTANT_InterfaceMethodref_info。

对方法句柄的引用来自于CONSTANT_MethodHandle_info。

对方法类型的引用来自于CONSTANT_MethodType_info。

对动态计算常量的符号引用来自于CONSTANT_MethodType_info。

对动态计算的call site的引用来自于CONSTANT_InvokeDynamic_info。

类加载器

类是怎么建立的呢?类的建立能够是由其余类调用该类的初始化方法来建立,也能够经过反射来建立。

类其实又能够分为两种,一种是数组类,一种是非数组类。

对于非数组类,由于他们有相应的二进制表示,因此是经过类加载器加载二进制表示来建立的。

而对于数组类,由于他们没有外部的二进制表示,因此数组类是由java虚拟机建立的。

java虚拟机中的类加载器又有两种,一种是虚拟机提供的引导类加载器,一种是用户自定义的类加载器。

若是是用户自定的类加载器,那么应该是ClassLoader的一个实现。用户自定义类加载器主要是为了扩展java虚拟机的功能,以支持动态加载并建立类。

连接

连接是为了让类或者接口能够被java虚拟机执行,而将类或者接口并入虚拟机运行时状态的过程。

连接具体的工做包括验证和准备类或者接口。而解析这个类或者接口中的符号引用是连接过程当中的可选部分。

若是java虚拟机选择在用到类或者接口中的符号引用时才去解析他们,这叫作延迟解析。

若是java虚拟机在验证类的时候就解析符号引用,这就叫作预先解析。

验证

验证主要是为了保证类和接口的二进制表示的结构正确性。

若是类或者接口的二进制表示不知足相应的约束,则会抛出VerifyError异常。

准备

准备主要是建立类或者接口的静态字段,并使用默认值来初始化这些字段。

解析

解析是指根据运行时常量池中的符号引用来动态决定其具体值的过程。

在执行java虚拟机指令:

anewarray,checkcat, getfield, getstatic, instanceof, invokedynamic, invokeinterface, invokespecial, invokestatic, invokevirtual, ldc, ldc_w, multianewarray, new , putfield和putstatic这些指令的时候,都会去将符号引用指向运行时常量池,从而须要对符号引用进行解析。

解析能够分为类和接口的解析,字段解析,普通方法的解析,接口方法解析,方法类型和方法句柄解析,调用点限定符解析这几种。

初始化

类或者接口的初始化是指执行类或者接口的初始化方法<clinit>。

只有下面的几种状况,类或者接口才会被初始化:

  1. 执行须要引用类或者接口的java虚拟机指令(new,getstatic, putstatic, invokestatic)的时候。
  2. 初次调用java.lang.invoke.Methodhandle实例的时候。
  3. 调用类库中的某些反射方法的时候。
  4. 对类的某个子类进行初始化的时候。
  5. 被选定为java虚拟机启动时候的初始类的时候。

总结

class文件通过加载,连接和初始化以后,就能够提供给JVM在运行时使用了。

本文做者:flydean程序那些事

本文连接:http://www.flydean.com/jvm-class-load-link-ini/

本文来源:flydean的博客

欢迎关注个人公众号:程序那些事,更多精彩等着您!

相关文章
相关标签/搜索