默认值 dcl new 替换 lan 语句块 一个 常见 静态变量
类从被加载到虚拟机内存中开始,到卸载出内存为止,它的整个生命周期包括:加载(Loading)、验证(Verification)、准备(Preparation)、解析(Resolution)、初始化(Initialization)、使用(Using)和卸载(Unloading)7个阶段。其中准备、验证、解析3个部分统称为连接(Linking)。如图所示。
加载、验证、准备、初始化和卸载这5个阶段的顺序是确定的,类的加载过程必须按照这种顺序按部就班地开始,而解析阶段则不一定:它在某些情况下可以在初始化阶段之后再开始,这是为了支持Java语言的运行时绑定(也称为动态绑定或晚期绑定)。以下陈述的内容都已HotSpot为基准。
在加载阶段(可以参考java.lang.ClassLoader的loadClass()方法),虚拟机需要完成以下3件事情:
加载阶段和连接阶段(Linking)的部分内容(如一部分字节码文件格式验证动作)是交叉进行的,加载阶段尚未完成,连接阶段可能已经开始,但这些夹在加载阶段之中进行的动作,仍然属于连接阶段的内容,这两个阶段的开始时间仍然保持着固定的先后顺序。
验证是连接阶段的第一步,这一阶段的目的是为了确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。
验证阶段大致会完成4个阶段的检验动作:
验证阶段是非常重要的,但不是必须的,它对程序运行期没有影响,如果所引用的类经过反复验证,那么可以考虑采用-Xverifynone参数来关闭大部分的类验证措施,以缩短虚拟机类加载的时间。
准备阶段是正式为类变量分配内存并设置类变量初始值的阶段,这些变量所使用的内存都将在方法区中进行分配。这时候进行内存分配的仅包括类变量(被static修饰的变量),而不包括实例变量,实例变量将会在对象实例化时随着对象一起分配在堆中。其次,这里所说的初始值“通常情况”下是数据类型的零值,假设一个类变量的定义为:
1
|
|
那变量value在准备阶段过后的初始值为0而不是123.因为这时候尚未开始执行任何java方法,而把value赋值为123的putstatic指令是程序被编译后,存放于类构造器()方法之中,所以把value赋值为123的动作将在初始化阶段才会执行。
至于“特殊情况”是指:public static final int value=123,即当类字段的字段属性是ConstantValue时,会在准备阶段初始化为指定的值,所以标注为final之后,value的值在准备阶段初始化为123而非0.
解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程。解析动作主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄和调用点限定符7类符号引用进行。
类初始化阶段是类加载过程的最后一步,到了初始化阶段,才真正开始执行类中定义的java程序代码。在准备极端,变量已经默认赋过系统要求的初始值,而在初始化阶段,是执行类构造器
1
2
3
4
5
6
7
8
9
|
|
由于父类的
接口中不能使用静态语句块,但仍然有变量初始化的赋值操作,因此接口与类一样都会生成
虚拟机会保证一个类的
虚拟机规范严格规定了有且只有5中情况(jdk1.7)必须对类进行“初始化”(而加载、验证、准备自然需要在此之前开始):
类的生命周期:装载、验证、准备、初始化、卸载有顺序,解析不保证严格顺序执行,验证可通过参数配置忽略提升类加载速度。
JVM没规定加载时机,但有初始化时机:用new构造函数或反射生成类对象时,调用类的静态变量或方法时,初始化类的子类时,类有main函数入口时。被动调用不会初始化类:调用类常量成员变量,调用父类静态成员。
JVM在初始化类时,会将其实例变量(成员变量)、实例代码块合并在父类构造函数后、构造函数代码区(init函数?)前执行,会将其静态成员变量、静态代码块合并成<cinit>函数在第一次初始化类时执行。
类变量在准备阶段分配内存空间并执行默认值操作,在生成一个对象时成员变量最多可以初始化四次:准备阶段分配内存空间并分配默认值,执行声明时赋值操作,执行实例代码块赋值操作,执行构造函数里的赋值操作。
Java类加载机制
来源: http://www.bubuko.com/infodetail-2319312.html