- 浏览: 172764 次
- 性别:
- 来自: 北京
文章分类
最新评论
-
GreatExpectations:
666可以可以哦
js并行加载,顺序执行 -
yiway:
如果是跨域的话,window.parent是拒绝访问的(由于w ...
利用HTML5的window.postMessage实现跨域通信 -
yiway:
如果是跨域的话,window.parent是决绝访问的(由于w ...
利用HTML5的window.postMessage实现跨域通信
详见: http://blog.yemou.net/article/query/info/tytfjhfascvhzxcytpo5
JVM:加载、链接和初始化
JVM要解释Java字节码,就必须对所需的类和接口执行如下3步操作:
(1) 加载:JVM在加载类时,会查找该类或该接口的二进制表示,并根据找到的二进制表示(通常是由Java编译器创建的类文件)创建一个Class对象。该Class对象中封装了类或接口的运行时状态。
由类加载器执行,该步骤将查找字节码,并从这些字节码中创建一个class对象。
(2) 链接:链接这一过程是指取得已加载的类或接口、结合JVM运行时环境、准备执行该类或该接口。
在链接阶段将验证类中的字节码,为静态域分配内存空间,并且如果必须的话,将解析这个类创建的对其他类的引用。
(3) 初始化:初始化是指JVM调用该类或该接口的初始化方法。
如果该类具有超类,则对其初始化,执行静态初始化器和静态初始化块。
初始化延迟到了对静态方法(构造器默认也是静态的)或者非常数静态域进行首次引用时才执行
1. 第一步
启动一个单机Java程序时,JVM首先做的是另外创建一个Class对象,用于表示包含public static void main(String [ ] args)方法的Java类。然后JVM会链接并初始化该Java类,调用main()方法,并用main()方法驱动所引用的其他类和接口的加载、链接和初始化过程。
2. 加载
加载过程是由类加载器完成的,该加载器是ClassLoader的子类,并且该类加载器会对所加载的类或接口进行一些校验检查。当表示已编译类或接口的二进制数据有错,则类或接口使用的类文件格式版本不被支持,类加载器找不到类或接口的定义,或者如果出现类循环,都会抛出异常。类循环是指类或接口的父类是其自身的情况。
类加载器一般有两种类型:由JVM提供的引导类加载器(bootstrap class loader)和用户定义的类加载器。用户定义的类加载器也是Java的ClassLoader类的子类,用于从非标准的、用户定义的源创建Class对象,以便提高安全性。例如,从加密文件中提取Class对象。一个加载器可以将部分甚至整个加载过程委托给另一个加载器。最终生成Class对象的加载器称为定义加载器(defining loader),而开始该加载过程的加载器称为启动加载器(initiating loader)。
使用默认引导类加载器的加载过程如下:根据所要加载的类文件,引导类加载器会判断自身是否已经成为该类的启动加载器。如果是,则Class对象存在,加载器停止(注意,加载一个类并不等于创建该类的一个实例,这一步骤仅仅是在JVM中加入该类)。如果类还没有加载,则加载器会搜索对应的类文件,并在找到后根据该类文件创建Class对象。如果找不到类文件,那么就会产生NoClassDefFoundError异常。
使用用户定义类加载器时,整个加载过程稍有不同。与引导加载器一样,用户定义的加载器首先判断自身是否已经成为目标类文件的启动加载器。如果是,则Class对象已经存在,加载器停止,而如果不是,用户定义的加载器会调用loadClass()方法。loadClass()方法返回所需的类文件并将表示类的二进制字节装配成ClassFile结构,然后调用defineClass()方法,由该方法从ClassFile结构创建Class对象。另外,loadClass()方法也可以将加载过程委托给另一个类加载器。
3. 链接
链接过程的第一步是校验需要链接的类文件。
Java类文件校验
由于JVM与Java编译器是完全分离的,因此,用来解释类文件的JVM无法保证类文件的形式正确,甚至无法保证该文件确实由Java编译器所生成。另一个问题在于继承与类兼容性。如果给定类文件所表示的类继承自另一个类文件表示的父类,那么JVM必须确保该子类的类文件与父类的类文件兼容。
JVM会校验每个类文件是否满足Java语言规范对类文件的约束,不过Java类校验器与Java语言无关。用某些其他语言编写的程序同样也能编译成类文件格式,编译之后,该类文件也能通过校验过程。
校验过程分为4个步骤:
(1) 第一步由JVM加载类文件并检查文件是否符合类文件的基本格式。类文件的长度必须准确。类文件必须确实表示类(检查其中一个特殊数字)。常量池中不能包含任何不可识别的信息,并且每个属性的长度正确。
(2) 校验过程的第二步在文件链接时进行。这一步执行的操作包括确保final关键字约束的保留。这表示final类不能派生子类,final方法也不能被重写。然后确保常量池中的元素符合Java语言的规定。验证常量池中的所有字段和方法引用,并检查每一个类(Object类除外)是否具有直接父类。
(3) 第三个校验步骤也在链接阶段进行。这一步检查类文件中引用的每一个方法,确保符合Java语言对方法的规定。方法调用中参数的数量和类型必须正确。操作数栈必须总保持相同大小,并包含相同类型的值。局部变量在访问前应当包含合适的值。必须为字段指定正确类型的值。
(4) 校验的最后一步是处理第一次调用方法时出现的事件,并保证一切按规范进行。这些检查包括:确保给定类中存在某个引用的字段或引用的方法,确认引用的字段或引用的方法具有正确的描述符,并确保一个方法在运行时能够访问该引用字段或引用方法。
准备
在校验类文件之后,JVM准备初始化类,包括为类变量分配内存空间并设置为默认初始值。这些值是标准的默认值,例如int类型为0,Boolean类型为false等。在初始化阶段,这些值会设为程序相关的默认值。
解析
在这一可选的步骤中,JVM把运行时常量池中引用的符号解析成具体值。
4. 初始化
链接过程完成后,会调用静态字段和静态初始化器。静态字段的值即使在类没有实例化时也能够访问得到,而静态初始化器用于单个表达式无法表示的静态初始化。JVM把所有这类初始化器收集到一个特殊的方法中。例如,类所有初始化器的集合就是初始化方法<clinit>。
不过,JVM在初始化一个类时不仅需要调用该类的初始化方法(只有JVM能够调用),而且需要初始化所有的父类(即需要调用这些父类的<clinit>)。结果就是,总是需要最先初始化Object类。另外,包含应用程序main()方法的类总是要初始化。
JVM:加载、链接和初始化
JVM要解释Java字节码,就必须对所需的类和接口执行如下3步操作:
(1) 加载:JVM在加载类时,会查找该类或该接口的二进制表示,并根据找到的二进制表示(通常是由Java编译器创建的类文件)创建一个Class对象。该Class对象中封装了类或接口的运行时状态。
由类加载器执行,该步骤将查找字节码,并从这些字节码中创建一个class对象。
(2) 链接:链接这一过程是指取得已加载的类或接口、结合JVM运行时环境、准备执行该类或该接口。
在链接阶段将验证类中的字节码,为静态域分配内存空间,并且如果必须的话,将解析这个类创建的对其他类的引用。
(3) 初始化:初始化是指JVM调用该类或该接口的初始化方法。
如果该类具有超类,则对其初始化,执行静态初始化器和静态初始化块。
初始化延迟到了对静态方法(构造器默认也是静态的)或者非常数静态域进行首次引用时才执行
1. 第一步
启动一个单机Java程序时,JVM首先做的是另外创建一个Class对象,用于表示包含public static void main(String [ ] args)方法的Java类。然后JVM会链接并初始化该Java类,调用main()方法,并用main()方法驱动所引用的其他类和接口的加载、链接和初始化过程。
2. 加载
加载过程是由类加载器完成的,该加载器是ClassLoader的子类,并且该类加载器会对所加载的类或接口进行一些校验检查。当表示已编译类或接口的二进制数据有错,则类或接口使用的类文件格式版本不被支持,类加载器找不到类或接口的定义,或者如果出现类循环,都会抛出异常。类循环是指类或接口的父类是其自身的情况。
类加载器一般有两种类型:由JVM提供的引导类加载器(bootstrap class loader)和用户定义的类加载器。用户定义的类加载器也是Java的ClassLoader类的子类,用于从非标准的、用户定义的源创建Class对象,以便提高安全性。例如,从加密文件中提取Class对象。一个加载器可以将部分甚至整个加载过程委托给另一个加载器。最终生成Class对象的加载器称为定义加载器(defining loader),而开始该加载过程的加载器称为启动加载器(initiating loader)。
使用默认引导类加载器的加载过程如下:根据所要加载的类文件,引导类加载器会判断自身是否已经成为该类的启动加载器。如果是,则Class对象存在,加载器停止(注意,加载一个类并不等于创建该类的一个实例,这一步骤仅仅是在JVM中加入该类)。如果类还没有加载,则加载器会搜索对应的类文件,并在找到后根据该类文件创建Class对象。如果找不到类文件,那么就会产生NoClassDefFoundError异常。
使用用户定义类加载器时,整个加载过程稍有不同。与引导加载器一样,用户定义的加载器首先判断自身是否已经成为目标类文件的启动加载器。如果是,则Class对象已经存在,加载器停止,而如果不是,用户定义的加载器会调用loadClass()方法。loadClass()方法返回所需的类文件并将表示类的二进制字节装配成ClassFile结构,然后调用defineClass()方法,由该方法从ClassFile结构创建Class对象。另外,loadClass()方法也可以将加载过程委托给另一个类加载器。
3. 链接
链接过程的第一步是校验需要链接的类文件。
Java类文件校验
由于JVM与Java编译器是完全分离的,因此,用来解释类文件的JVM无法保证类文件的形式正确,甚至无法保证该文件确实由Java编译器所生成。另一个问题在于继承与类兼容性。如果给定类文件所表示的类继承自另一个类文件表示的父类,那么JVM必须确保该子类的类文件与父类的类文件兼容。
JVM会校验每个类文件是否满足Java语言规范对类文件的约束,不过Java类校验器与Java语言无关。用某些其他语言编写的程序同样也能编译成类文件格式,编译之后,该类文件也能通过校验过程。
校验过程分为4个步骤:
(1) 第一步由JVM加载类文件并检查文件是否符合类文件的基本格式。类文件的长度必须准确。类文件必须确实表示类(检查其中一个特殊数字)。常量池中不能包含任何不可识别的信息,并且每个属性的长度正确。
(2) 校验过程的第二步在文件链接时进行。这一步执行的操作包括确保final关键字约束的保留。这表示final类不能派生子类,final方法也不能被重写。然后确保常量池中的元素符合Java语言的规定。验证常量池中的所有字段和方法引用,并检查每一个类(Object类除外)是否具有直接父类。
(3) 第三个校验步骤也在链接阶段进行。这一步检查类文件中引用的每一个方法,确保符合Java语言对方法的规定。方法调用中参数的数量和类型必须正确。操作数栈必须总保持相同大小,并包含相同类型的值。局部变量在访问前应当包含合适的值。必须为字段指定正确类型的值。
(4) 校验的最后一步是处理第一次调用方法时出现的事件,并保证一切按规范进行。这些检查包括:确保给定类中存在某个引用的字段或引用的方法,确认引用的字段或引用的方法具有正确的描述符,并确保一个方法在运行时能够访问该引用字段或引用方法。
准备
在校验类文件之后,JVM准备初始化类,包括为类变量分配内存空间并设置为默认初始值。这些值是标准的默认值,例如int类型为0,Boolean类型为false等。在初始化阶段,这些值会设为程序相关的默认值。
解析
在这一可选的步骤中,JVM把运行时常量池中引用的符号解析成具体值。
4. 初始化
链接过程完成后,会调用静态字段和静态初始化器。静态字段的值即使在类没有实例化时也能够访问得到,而静态初始化器用于单个表达式无法表示的静态初始化。JVM把所有这类初始化器收集到一个特殊的方法中。例如,类所有初始化器的集合就是初始化方法<clinit>。
不过,JVM在初始化一个类时不仅需要调用该类的初始化方法(只有JVM能够调用),而且需要初始化所有的父类(即需要调用这些父类的<clinit>)。结果就是,总是需要最先初始化Object类。另外,包含应用程序main()方法的类总是要初始化。
发表评论
-
博客已转移至 http://blog.yemou.net/
2015-07-27 17:57 879我所有的博客都已经转移至 http://blog.yemo ... -
java对象 深度克隆(不实现Cloneable接口)和浅度克隆
2015-06-08 15:45 1005详见: http://blog.yemou.net/art ... -
java 多线程Callable和Runable执行顺序问题详解
2015-02-28 16:25 1094详见: http://blog.yemou.net/art ... -
java 读写锁详解
2015-02-26 17:46 983详见: http://blog.yemou.net/art ... -
jvm 常用内存分析命令
2015-02-06 15:32 730详见: http://blog.yemou.net/art ... -
Spring AOP 详解
2015-01-15 15:03 770详见: http://blog.yemou.net/art ... -
正确理解ThreadLocal
2014-06-17 16:54 676详见: http://blog.yemou.n ... -
探秘Java虚拟机——内存管理与垃圾回收
2014-06-16 23:29 1185详见: http://blog.yemou. ... -
Java GC 日志详解
2014-06-16 23:11 797详见: http://blog.yemou.net/art ... -
垃圾回收调优及JVM参数详解
2014-06-13 09:09 647http://hi.baidu.com/jiangyangw3 ... -
深入理解Java HashMap实现原理
2014-04-10 18:05 740详见: http://blog.yemou.net/art ... -
【转】Java HashMap的死循环问题
2014-04-10 17:12 912详见: http://blog.yemou.net/art ... -
HashMap和ConcurrentHashMap对null的不同处理
2014-04-10 15:45 848详见: http://blog.yemou.net/art ... -
探索 ConcurrentHashMap 高并发性的实现机制
2014-03-31 17:54 848很不错的一篇文章,值得一看 http://www.ibm ... -
并发队列ConcurrentLinkedQueue和阻塞队列LinkedBlockingQueue用法
2014-03-25 22:28 728并发队列ConcurrentLinkedQueue和阻塞队列 ... -
[转]Java中的随机数生成器:Random,ThreadLocalRandom,SecureRandom
2014-03-25 22:08 1376详见: http://blog.yemou ... -
[转]Java7中的ForkJoin并发框架初探(下)—— ForkJoin的应用
2014-03-19 15:47 578详见: http://blog.yemou.net/art ... -
[转]Java7中的ForkJoin并发框架初探(中)——JDK中实现简要分析
2014-03-19 15:46 737详见: http://blog.yemou.net/art ... -
[转]Java7中的ForkJoin并发框架初探(上)——需求背景和设计原理
2014-03-19 15:44 760详见: http://blog.yemou.n ... -
[转] Java se 7新特性研究(二)
2014-03-19 11:38 1380详见: http://blog.yemou.net/art ...
相关推荐
JVM类加载过程 JVM(Java Virtual Machine)是Java语言的核心组件之一,它是Java语言的可移植性和跨平台性的基础。JVM主要组成部分包括类加载子系统、执行引擎、本地方法接口和运行时数据区。 类加载子系统是JVM的...
**案例背景**:本案例通过编写一个简单的Java程序,使用单例模式初始化静态变量和对象,来演示类的加载过程以及类加载器的委托机制。 **案例源码**: ```java public class Singleton { private static Singleton...
类加载过程中可能会遇到一些问题,如类冲突、类循环依赖等。类冲突通常发生在不同加载器加载了相同包名下的类,导致版本不一致。类循环依赖则可能引发无限递归,使得类加载失败。解决这些问题通常需要对类加载器的...
类的加载过程包括三个主要步骤:加载、连接和初始化。 1. 加载:读取.class文件并生成对应的Class对象。这个过程可能涉及文件查找、解压等操作。 2. 链接:分为验证、准备和解析三个阶段。 - 验证:确保被加载的...
JVM参数配置,运行时数据区
总结,Java 类加载器是JVM中的重要组成部分,它决定了类的加载过程和加载源,双亲委派机制保证了类加载的有序性和安全性。理解类加载器的工作原理有助于我们更好地进行程序设计和优化,特别是在开发插件系统、模块化...
java JVM 类加载-初始化 过程
加载过程中,字节码被转换为内存中的数据结构,并在方法区(在JVM规范的早期版本中称为永久代,现代JVM中称为元空间)中存储。同时,会在堆中创建一个`java.lang.Class`对象,作为方法区数据的封装。 2. 连接(Link...
双亲委派模型的工作流程是:从顶层的Bootstrap ClassLoader开始,逐级向下尝试加载类,直到找到能加载的类为止。 通过深入JDK源码,我们可以更清晰地理解类加载机制的细节,从而在遇到问题时能迅速定位并解决。对于...
被动使用则是在JVM运行过程中,由于JVM或JDK内部需要而进行的加载,如JNI查找类、JDK工具类的使用等。 在Java开发中,理解JVM的类加载机制对于优化性能、解决类冲突问题、实现自定义类加载器等都有重要意义。例如,...
Java虚拟机(JVM)的类加载机制是Java运行时环境的重要组成部分,它负责将类的字节码文件加载到内存中,进行一系列处理并使其成为可执行的Java类型。这个过程包括五个主要阶段:加载、验证、准备、解析和初始化。 1...
本专题"性能调优专题-jvm类加载机制-performance-jvmclassloader"深入探讨了如何通过理解并优化类加载过程来提升应用程序的性能。 1. **类加载器的层次结构** Java中的类加载器分为Bootstrap ClassLoader、...
#### 三、类加载过程详解 类加载的过程主要包括以下几个步骤: **3.1 加载** 在这个阶段,JVM根据类的全限定名读取对应的二进制数据流,并将其转换为JVM内部表示的`Class`对象。 **3.2 验证** 验证阶段是为了...
通过对Java中JVM加载`.class`文件的过程及其类加载器的具体工作原理的介绍,我们可以更加深刻地理解Java程序的运行机制。了解这些基础知识对于开发高质量的Java应用程序至关重要。通过掌握类加载器的工作方式,...
本文主要探讨JVM的类加载机制,包括类加载、连接、初始化等关键过程,以及类的主动使用和被动使用的情况。 首先,我们要理解**类加载**的作用。JVM的类加载器(ClassLoader)负责将编译后的`.class`文件加载到内存...
首先,我们要知道Java程序的类加载过程通常分为三个阶段:加载、验证、准备、解析和初始化。其中,加载阶段就是JVM查找并解析字节码文件的过程。Java类的加载主要由类加载器完成,它分为引导类加载器(Bootstrap ...
当遇到`java.lang.ClassNotFoundException`异常时,通常是因为类加载过程出现了问题。了解类加载机制对于解决这类问题至关重要,同时也有助于深入理解Java虚拟机(JVM)的工作原理。 Java 类加载机制主要由类加载器...
类加载器在类加载过程中扮演重要角色,它负责找到类的字节流并将其转换为运行时可用的形式。系统提供了几个内置类加载器,如Bootstrap ClassLoader(引导类加载器)、Extension ClassLoader(扩展类加载器)和...