Java虚拟机类加载机制

Wesley13
• 阅读 658

概述

  虚拟机把描述类的数据从Class文件加载到内存,并对数据进行校验、转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的类加载机制。

  与那些在编译时需要进行连接工作的语言不同,在Java语言里面,类型的加载、连接和初始化过程都在程序运行期间完成的,这种策略虽然会稍微增加一些系统性能开销,但是会为Java应用程序提供高度的灵活性,而Java在运行时期动态加载和动态连接这个两个

  特点正式Java可以动态拓展的特性。

  我们可以通过Java预定义的和自定义类加载器,让一个存在的本地程序在运行时从网络或者其他地方加载一个二进制流作为程序的一部分,例如最基础的Applet、JSP、OSGI技术等,都是利用了Java语言运行期类加载的特性。

类加载的时机

  类从被加载到虚拟机内存中开始,到卸载出内存为止,它的整个生命周期包括:加载(Loading)、验证(Verification)、准备(Preparation)、解析(Resolution)、初始化(Initialization)、使用(Using)、和卸载(Unloading)7个阶段。

  其中验证、准备、解析3个部分统称为连接(Linking)。

  加载、验证、准备、初始化、和卸载这个5个阶段的顺序时确定的,类的加载过程必须按照这种顺序按部就班的开始,而解析阶段则不一定,通常会在一个阶段执行的过程中调用、激活另外一个阶段,在某些情况下可以在初始化之后再解析,这是为了支持Java

  语言的运行时绑定(也叫动态绑定或者晚期绑定)。

  Java虚拟机并没有强制约束何时开始加载,但是对于一下5中情况必须对类进行初始化有强制规定。

  1. 遇到new、getstatic、putstatic、和invokestatic这4条字节码指令时,如果类没有进行过初始化,则需要先触发其初始化。而生存这四条指令最常见的Java代码场景是:使用new关键字实例化对象的时候、读取或设置一个类的静态字段(被final修饰、已在编译器就把结果放入常量池的静态字段除外)的时候,以及调用一个类的静态方法的时候。
  2. 使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行过初始化,则需要先触发初始化
  3. 当初始化一个类的时候,如果发现它的父类还没有进行过初始化,则需要先触发其父类的初始化
  4. 当虚拟机启动的时候,用户需要指定一个要执行的主类(包含main()方法的那个类),虚拟机会先初始化这个主类
  5. 当使用jdk1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后解析的结果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,并且这个方法句柄所对应的类没有进行过初始化,则需要先触发其初始化

  对于这5种会触发类进行初始化的场景,可以称为对一个类进行主动引用。除此之外,所有引用类的方式都不会触发初始化,称为被动引用。

  实例代码

public class SuperClass {
        
    static {
        System.out.println("SuperClass init!");
    }
    
    public static int value = 123;    
}


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

public class TestMain {

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

}

运行结果
SuperClass init!
123

为什么只会输出SuperClass init! 而不会输出SubClass init!。因为对于静态字段,只有直接定义这个字段的类才会被初始化,因此通过其子类来引用父类种定义的静态字段,只会触发父类的初始化而不会触发子类的初始化。对于Sun HotSpot虚拟机来说,可以通过-XX:+TraceClassLoading参数观察到此操作会导致子类的加载。

例子二

public class TestMain {

    public static void main(String[] args) {
        //System.out.println(SubClass.value);
        SuperClass [] scaClasses = new SuperClass[10];
    }

}运行结果:无

为什么没有输出SuperClass init!呢,说明并没有触发SuperClass的初始化阶段。

类加载的过程

  1. 通过一个类的全限定名来获取定义此类的二进制字节流。
  2. 将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构。
  3. 在内存种生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据的访问入口。
点赞
收藏
评论区
推荐文章
待兔 待兔
4个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Wesley13 Wesley13
3年前
java基础知识随身记
2018年11月12日20:51:35一、基础知识:1、JVM、JRE和JDK的区别:JVM(JavaVirtualMachine):java虚拟机,用于保证java的跨平台的特性。  java语言是跨平台,jvm不是跨平台的。JRE(JavaRuntimeEnvironment):java的运行环境,包括jvmjava的核心类
Wesley13 Wesley13
3年前
Java高级篇——深入浅出Java类加载机制
类加载器简单讲,类加载器ClassLoader的功能就是负责将class文件加载到jvm内存。类加载器分类从虚拟机层面讲分为两大类型的类加载器,一是BootstrapClassloader即启动类加载器(C实现),它是虚拟机的一部分,二是其他类型类加载器(JAVA实现),在虚拟机外部,并全部继
Wesley13 Wesley13
3年前
Java日期时间API系列31
  时间戳是指格林威治时间1970年01月01日00时00分00秒起至现在的总毫秒数,是所有时间的基础,其他时间可以通过时间戳转换得到。Java中本来已经有相关获取时间戳的方法,Java8后增加新的类Instant等专用于处理时间戳问题。 1获取时间戳的方法和性能对比1.1获取时间戳方法Java8以前
Stella981 Stella981
3年前
KVM调整cpu和内存
一.修改kvm虚拟机的配置1、virsheditcentos7找到“memory”和“vcpu”标签,将<namecentos7</name<uuid2220a6d1a36a4fbb8523e078b3dfe795</uuid
Easter79 Easter79
3年前
Twitter的分布式自增ID算法snowflake (Java版)
概述分布式系统中,有一些需要使用全局唯一ID的场景,这种时候为了防止ID冲突可以使用36位的UUID,但是UUID有一些缺点,首先他相对比较长,另外UUID一般是无序的。有些时候我们希望能使用一种简单一些的ID,并且希望ID能够按照时间有序生成。而twitter的snowflake解决了这种需求,最初Twitter把存储系统从MySQL迁移
Stella981 Stella981
3年前
Jvm类的加载机制
1.概述虚拟机加载Class文件(二进制字节流)到内存,并对数据进行校验、转换解析和初始化,最终形成可被虚拟机直接使用的Java类型,这一系列过程就是类的加载机制。2.类的加载时机类从被虚拟机加载到内存开始,直到卸载出内存为止,整个生命周期包括:加载——验证——准备——解析——初始化——使用——卸载这7个阶段。其中验
Wesley13 Wesley13
3年前
JDBC之数据库的连接步骤(六步)
1.加载驱动在连接数据库之前,需要加载数据库的驱动到JVM(Java虚拟机),这需要通过java.lang.Class类的静态方法forName(StringclassName)实现.例如://加载Oracle的驱动try{Class.forName("oracle.jdbc.OracleDriver");}catch(ClassNo
Stella981 Stella981
3年前
Android动态加载之ClassLoader详解
Dalvik虚拟机如同其他Java虚拟机一样,在运行程序时首先需要将对应的类加载到内存中。而在Java标准的虚拟机中,类加载可以从class文件中读取,也可以是其他形式的二进制流。因此,我们常常利用这一点,在程序运行时手动加载Class,从而达到代码动态加载执行的目的。只不过Android平台上虚拟机运行的是Dex字节码,一种对class文件优化的产物
Stella981 Stella981
3年前
JVM(四)JVM的双亲委派模型
1、两种不同的类加载器  从JAVA虚拟机的角度来讲,只存在两种不同的类加载器:一种是启动类加载器(BootstrapClassLoader),这个类加载器使用C语言实现,是虚拟机自身的一部分;另一种就是所有其他的类加载器,这些加载器都由Java语言实现,独立于虚拟机外部,并且全都继承自抽象类java,lang.ClassLoader。