文档章节

虚拟机类加载机制概叙1:类加载的时机

玄影镜心
 玄影镜心
发布于 2015/06/16 11:24
字数 1034
阅读 13
收藏 0

    在java语言里,类型的加载、连接和初始化过程都是在程序运行期间完成的(增加了开销,但是更加灵活,动态扩展就是依赖与此)

一:生命周期

    类的生命周期:加载----验证----准备----解析----初始化----使用----卸载      (7个阶段)

    其中:验证、准备、解析3个部分统称为连接阶段

    顺序加载、验证、准备、初始化和卸载,这五个阶段顺序是确定的,类加载过程中严格遵守顺序。而解析阶段在某些情况下可以在初始化之后再开始(为了支持JAVA的动态绑定)


二:什么时候初始化?

    虚拟机规范严格规定了,有且只有5种情况必须立即对类进行“初始化”(而加载、验证、准备自然需要在此之前开始[严格的顺序])。

1:遇到new、getstatic、putstatic或者invokestatic这4条字节码命令时,如果类没有初始化,则触发
2:使用java.lang.reflect包的方法对类进行反射调用时,如果类没有初始化,则触发
3:当初始化一个类时没,如果父类没有初始化没,则先出发其父类初始化
4:当虚拟机启动时,用户需要指定一个要执行的主类(含有main方法的类),虚拟机会先初始化这个主类
5:当使用JDK1.7的动态语言时,如果一个MethodHandle实例最后的解析结果是REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,并且这个方法句柄对应的类没有初始化,则需要先出发其初始化。

说明:

    1:第一条的使用场景,使用new关键字实例化对象的时候、读取或设置一个类的静态字段时候(被final修饰、在编译期把结果放入常量池的静态字段除外)、调用一个类的静态方法的时候。

    2:除了以上5条之外,所有引用类的方法都不会触发初始化。(被动引用)

代码清单1:被动引用例子一

//初始化ConstClass时会先触发SuperClass的初始化,如果main方法放在ConstClass中,则先初始化ConstClass
//因为含有main方法的主类会被虚拟机优先初始化。  
public class SuperClass {
	static{
		System.out.println("SpuerClass init!");
	}	
	public static int CCC = 123;
}

public class ConstClass extends SuperClass {
	static{
		System.out.println("ConstClass init!");
	}
}

public class Test{
    public static void main(String[] args){
        System.out.println(ConstClass.value);
    }
}

//打印结果:SpuerClass init!
//         123

    从子类调用父类静态变量,子类并未被初始化。由此可见,对于静态字段的调用,只有直接定义这个字段的类才会被初始化。

代码清单2:被动引用例子二

//通过数组定义来引用类,不会触发此类的初始化
public class Test{
    public static void main(String[] args){
        SuperClass[] as = new SuperClass[10];
    }
}

SuperClass并没有被初始化,但是这个段代码触发了另一个名为“Lorg.fenixsoft.classloading.SuperClass”的类的初始化,这是由虚拟机自动生成的、创建动作由newarray指令触发。

代码清单3:被动引用例子三

//SuperClass不会被初始化
public class SuperClass {
	static{
		System.out.println("SpuerClass init!");
	}
	public static final String HELLO = "hello";
}

public class Test{
    public static void main(String[] args){
        System.out.println(SuperClass.HELLO);
    }
}

    常量在编译阶段会存入调用类的常量池中,本质上并没有直接引用到定义常量的类,因此不会触发这个类的初始化。对HELLO的引用被直接放到Test类的常量池中。


三:接口和类加载的不同点:

    接口与类的加载,真正有所区别的是前面有且只有5种场景的第三种:一个接口在初始化时,并不要求父类接口全部都完成了初始化,只有在真正使用到父接口的时候才会初始化。

© 著作权归作者所有

共有 人打赏支持
玄影镜心
粉丝 8
博文 90
码字总数 49460
作品 0
西安
高级程序员
(八)-虚拟机的类加载机制

虚拟机把描述类的数据从Class文件加载到内存,并对数据进行校验、转换解析和初始化,最终形成可以被虚拟机直接使用的Java类. 即虚拟机的类加载机制. 与那些在编译时需要进行链接工作的语言不同...

芥末无疆sss
2017/12/19
0
0
java中static{}语句块详解

原文地址:http://blog.csdn.net/lubiaopan/article/details/4802430 感谢原作者! static{}(即static块),会在类被加载的时候执行且仅会被执行一次,一般用来初始化静态变量和调用静态方法,...

巴顿
2014/01/20
0
0
JVM详解之类加载过程

JVM详解之类加载过程 什么叫类?类在什么时候进行加载?类加载过程发生了什么?或者叫jvm做了哪些工作? 类的概念(.java/.class的区别)和范围扩充 java语言和Java虚拟机的概念 java可以运行...

默默学习中
2016/04/15
575
0
深入理解JVM类加载机制

简述:虚拟机把描述类的数据从class文件加载到内存,并对数据进行校验、转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的类加载机制。 下面我们具体来看类加载的过...

Shuqing,Wang
2017/11/29
0
0
Java 中的 GC -- GC基础常识

---------- 文中讨论的GC原理均基于 Sun Hotspot JVM,对于不同 JVM 实现可能会有不同。 1. GC (Gabage Collector) 做了什么事情 决定哪些内存需要回收(物理内存的位置)。 决定何时回收这...

longjuelegend
2016/08/30
26
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

Zookeeper总结

Zookeeper的部分概念 什么是zookeeeper? Zookeeper是一个分布式服务的协调中心 zookeeper节点的角色类型? Leader(领导者)、Follower(跟随者)、Observer(观察者) Leader 负责更新系统...

DemonsI
22分钟前
1
0
Redis学习笔记

常用命令 从Docker进入Redis的命令 sudo docker exec -it redis /bin/bash

OSC_fly
23分钟前
0
0
SqlServer查询某个日期的数据

select * from View_ZJMONITORINGCORROSION where ENTERDATE > CONVERT(datetime,DATEADD(day,1,'2017/12/28 14:53:07'))...

笑丶笑
24分钟前
0
0
常用编码规范

Standard characters https://ascii.cl/

yeahlife
26分钟前
0
0
flannel实战

docker swarm mode的出现是个里程碑,官方原生的编排调度看起来都成雏形了,但是swarm mode和容器外部系统的对接、网络性能始终不尽人意,swarm mode下各种开源周边不能使用,感觉swarm mod...

China_OS
27分钟前
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部