类加载过程
也就是加载、验证、准备、解析、初始化这5阶段的具体动作:
加载阶段
①通过一个类的全限定名来获取定义此类的二进制字节流
②将这个字节流所代表的静态存储结构转换为方法区的运行时数据结构
③在内存中生成一个代表这个类的 java.lang.Class 对象,作为方法区这个类的各种数据的访问入口.
虚拟机规范并没有详细说明从哪里获取一个二进制字节流,充满创造力的开发人员玩出了各种花样,例如从 zip 包中读取,最后从 jar、ear、war 中读取.从网络中读取,运行时计算生成.由其他文件生成,例如 jsp
相对于类加载过程的其他阶段,一个非数组类的加载阶段(准确的说是加载阶段中获取类的二进制字节流的动作)是开发人员可控性最强的,因为加载阶段既可以使用系统提供的引导类加载器来完成,开发人员可以通过定义自己的类加载器去控制字节流的获取方式(即重写一个类加载器的 loadClass 方法).
对于数组类而言,情况有所不同,数组类本身不是通过类加载器创建,它是由 java 虚拟机直接创建的. 但数组类和类加载器还是有很大联系的,数组类的元素类型最终还是要靠类加载器去创建,一个数组类创建过程遵循以下规则:
1.如果数组的组件类型(Component Type,指的是数组去掉一个维度的类型)是引用类型,那就递归采用本届中定义的加载过程去加载这个组件类型,数组C将在加载该组件类型的类加载器的类名称空间上被标识
2.如果数组的组件类型不是引用类型(例如 int[] 数组),java虚拟机将会把数组C标记为与引导类加载器关联
3.数组类的可见性与它的组件类型的可见性一致,如果组件类型不是引用类型,那么数组类的可见性将默认为 public.
加载阶段完成后,虚拟机外部的二进制字节流就按照虚拟机所需的格式存储在方法区中,方法区中的数据存储格式由虚拟机实现自行定义.然后在内存中实例化一个 java.lang.Class 类的对象(并没有明确规定是在 java 堆中,对于 Hotspot 虚拟机而言,Class 对象比较特殊,它虽是一个对象,但是存放在方法区里),这个对象作为程序访问方法区中的这些类型数据的外部接口.
加载阶段与连接阶段的部分内容(如一部分字节码文件格式验证动作)是交叉进行的,加载阶段尚未完成,连接阶段可能已经开始.
验证阶段
这一阶段是为了确保 Class 文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全.
这一阶段很重要,直接决定了 Java 虚拟机是否能承受恶意代码的攻击,但是虚拟机规范中没有给出具体检查哪些方面,如何检查,何时检查,都没有足够具体的要求和明确的说明. 直到2011年发布的 java SE 7. 从整体上看,验证阶段大致会完成下面4个阶段的检查动作:文件格式验证、元数据验证、字节码验证、符号引用验证.
1.文件格式验证:验证字节流是否符合 Class 文件格式的规范,并且能被当前版本的虚拟机处理.
①是否以魔数 0xCAFEBABE 开头
②主、次版本号是否在当前虚拟机处理范围内
③常量池中的常量是否有不被支持的常量类型
④指向常量池的各种索引值中是否有指向不存在的常量或不符合类型的常量
⑤Constant_Utf8_info 型的常量中是否有不符合 UTF8 编码的数据
⑥Class文件中各个部分及文件本身是否有被删除的或附加的其他信息
该验证阶段的主要目的是要保证输入的字节流能正确的被解析并存储在方法区之内,格式上符合描述一个 java 类型信息的要求. 后面 3 个验证阶段全部是基于方法区的存储结构进行,不会再直接操作字节流.
2.元数据验证
第二阶段是对字节码描述的信息进行语义分析,以保证其描述的信息符合 java 语言规范的要求,这一阶段可能包括的验证点如下:
①这个类是否有父类(除了 Object 外,其他所有的类都有父类)
②这个类的父类是否继承了不允许被继承的类(被 final 修饰的类)
③如果这个类不是抽象类,是否实现了其父类或接口中要求的所有方法
④类中的字段、方法是否与父类产生矛盾(例如覆盖了父类的 final 字段)
第二阶段的主要目的是对类的元数据信息进行语义校验,保证不存在不符合 java 语言规范的元数据信息。
3.字节码验证
主要目的是通过数据流和控制流分析,确保程序语义是合法的、符合逻辑的, 这个阶段对类的方法体进行校验分析,确保被校验类的方法在运行时不会做出危害虚拟机安全的事件,例如:
①保证任何时刻操作数栈的数据类型与指令代码序列都能配合工作
②保证跳转指令不会跳转到方法体以外的字节码指令上
③保证方法体中的类型转换是有效的
4.符号引用验证
最后一个阶段的校验发生在虚拟机将符号引用转换为直接引用的时候,这个转换动作将在了解的第三阶段——解析阶段中发生. 符号引用验证可以看做是对类自身以外(常量池中的各种符号引用)的信息进行匹配性校验,通常需要校验下列内容:
①符号引用中通过字符串描述的全限定名是否能找到对应的类
②在指定类中是否存在符合方法的字段描述符以及简单名称所描述的方法和字段
③符号引用中的类、字段、方法的访问性是否可被当前类访问.
符号引用验证的目的是确保解析动作能正常执行,如果无法通过符号引用验证,那么将会抛出 java.lang.IncompatibleClassChangeError 异常的子类.
验证阶段并不是一定需要的,如果所有运行的全部代码都已被反复使用和验证过,那么在实施阶段就可以考虑使用 -Xverify:none 参数来关闭大部分的类验证措施,以缩短虚拟机类加载的时间.
准备阶段
准备阶段是正式为类变量分配内存并设置类变量初始值的阶段,这些变量所使用的内存都将在方法区中进行分配. 值得注意的是,这里只是为类变量分配内存,而不包括实例变量,实例变量在堆中分配内存. 再者,这里说的初始值通常情况下是该数据类型的零值,假设:
public static int value = 123;
那么变量 value 在准备阶段过后的初始值是 0 而不是 123.
需要注意的是,如果类字段的字段属性表中存在 ConstantValue 属性,那么在准备阶段变量 value 就会被初始化为 ConstantValue 属性所指定的值.
public static final int value = 123;
分享到:
相关推荐
什么是虚拟机类加载机制以及加载过程,以及类加载时机
Java虚拟机(JVM)的类加载过程是Java程序运行的基础,它涉及到类加载器、类的生命周期和一系列复杂的步骤。在这个过程中,类加载器主要任务是根据类的全限定名加载二进制字节流并转化为`java.lang.Class`对象。整个...
虚拟机将描述类的数据从Class文件加载到内存,并对数据进行校验、准备、解析和初始化,终会形成可以被虚拟机使用的Java类型,这是一个虚拟机的类加载机制。Java中的类是动态加载的,只有在运行期间使用到该类的...
加载过程中,字节码被转换为内存中的数据结构,并在方法区(在JVM规范的早期版本中称为永久代,现代JVM中称为元空间)中存储。同时,会在堆中创建一个`java.lang.Class`对象,作为方法区数据的封装。 2. 连接(Link...
方法区也被称为永久代(PermGen space),主要用来存储已被虚拟机加载的类信息、常量、静态变量以及即时编译器编译后的代码等数据。值得注意的是,不同虚拟机实现对方法区的具体布局可能有所不同,这些差异取决于...
类加载的生命周期包括加载、验证、准备、解析、初始化、使用和卸载七个阶段,其中加载、验证、准备、初始化和卸载这五个阶段是确定的,类的加载过程必须按照这种顺序进行。 加载阶段是将.class文件加载到内存中,...
#### 三、类加载过程详解 类加载的过程主要包括以下几个步骤: **3.1 加载** 在这个阶段,JVM根据类的全限定名读取对应的二进制数据流,并将其转换为JVM内部表示的`Class`对象。 **3.2 验证** 验证阶段是为了...
下面我们将详细探讨WebLogic的类加载过程。 1. **类加载器层次结构** WebLogic的类加载器体系遵循“父类加载器优先”原则,这意味着当一个类加载请求发生时,首先由父类加载器尝试加载,只有在其无法找到相应类时...
类加载过程包括通过一个类的全限定名来获取定义此类的二进制字节流,将这个字节流所代表的静态存储结构转换为方法区的运行时数据结构,在内存中生成一个代表整个类的java.lang.Class对象,作为方法区这个类的各种...
本文主要探讨JVM的类加载机制,包括类加载、连接、初始化等关键过程,以及类的主动使用和被动使用的情况。 首先,我们要理解**类加载**的作用。JVM的类加载器(ClassLoader)负责将编译后的`.class`文件加载到内存...
Java 类加载过程是Java程序运行时的关键环节,它涉及到如何将类的字节码加载到JVM中并准备执行。整个过程可以分为五个主要步骤:加载(Loading)、验证(Verification)、准备(Preparation)、解析(Resolution)和...
首先,让我们了解类加载的基本过程。当JVM启动时,会触发类加载。这个过程分为三个阶段:加载、链接和初始化。加载阶段,类加载器找到对应的.class文件;链接阶段,字节码被校验并转化为内存中的数据结构;初始化...
【Java虚拟机加载Java类的过程】 Java虚拟机(JVM)加载Java类的过程是一个关键的运行时机制,涉及三个主要步骤:加载、链接和初始化。这个过程确保了类的正确构造和执行。以下是对这些步骤的详细解释: 1. **加载...
二、类加载过程 类加载过程包括三个主要阶段:加载、验证、准备、解析和初始化。 1. 加载:类加载器读取类的二进制数据,这些数据通常来自.class文件,也可能来自网络、数据库或其他数据源。 2. 验证:确保加载的...
类加载过程中可能会遇到一些问题,如类冲突、类循环依赖等。类冲突通常发生在不同加载器加载了相同包名下的类,导致版本不一致。类循环依赖则可能引发无限递归,使得类加载失败。解决这些问题通常需要对类加载器的...
类加载器在类加载过程中扮演重要角色,它负责找到类的字节流并将其转换为运行时可用的形式。系统提供了几个内置类加载器,如Bootstrap ClassLoader(引导类加载器)、Extension ClassLoader(扩展类加载器)和...
Java虚拟机类装载机制是Java运行环境中的核心组成部分,它负责将类的字节码从磁盘、网络等不同来源加载到JVM中,并进行一系列处理以使类能够被正确地使用。类装载机制的目的是为了实现代码的动态加载和运行时的灵活...
5. **类加载与DexOpt**:在运行Java代码之前,Dalvik虚拟机需要加载和解析类。这涉及到dex文件(Dalvik Executable Format)的读取,以及可能的优化过程(DexOpt),优化后的.dex文件会被存储在设备的缓存区域以提高...
类加载过程涉及到类的加载、链接(验证、准备、解析)、初始化等阶段,并且这一过程是由类加载器系统完成的。 #### 二、类加载器系统 Java中的类加载器系统主要包括以下几种类型的类加载器: 1. **Bootstrap ...