JVM中类的加载
1、类的加载过程
1.1、加载
加载是指类加载器将类的class文件读入到内存中,并为之创建一个Class对象。
1.2、链接
类被加载之后,系统生成一个对应的Class对象,接着将会进入链接阶段,链接阶段负责把类的二进制数据合并到JRE中。链接阶段可以细分为下面三个阶段:
- 验证: 检验被加载的类是否有正确的内部结构,确保class文件的字节流中包含的信息符合当前虚拟机的要求,不会危害虚拟机的自身安全,主要包含四种验证:文件格式验证、元数据验证、字节码验证、符号引用验证,并和其他类协调一致。
- 文件格式验证:主要验证字节流是否符合class文件规范,并且能被当前的虚拟机加载处理。例如:常量池中是否有不被支持的常量类型,指向常量的索引值是否存在不存在的常量或不符合类型的常量。
- 元数据验证:对字节码描述信息进行语义分析,是否符合Java语言语法规范。
- 字节码验证:最重要的验证环节,分析数据流的控制,确定语义是合法的,符合逻辑的。主要针对元数据验证后对方法体的验证,保证方法在运行时不会出现危害。
- 符号引用验证:主要是针对符号引用转换为直接引用时候,是会延伸到第三解析阶段,主要确定访问类型等涉及到引用的情况,保证引用一定会被访问到,不会出现类等无法访问的情况。
- 准备:负责为类的静态变量分配内存,并设置默认初始值。
- 解析:将类的二进制数据中的符号引用替换成直接引用。
1.3、初始化
初始化是为类的静态变量赋予正确的初始值,准备阶段和初始化阶段看似有点矛盾,其实是不矛盾的,如果类中有语句private static int num = 5,它的执行过程是这样的,首先字节码文件被加载到内存后,先进行链接验证这一步,验证通过后进入准备阶段,给num分配内存,因为变量num是static的,所以此时int类型的num默认初始值为0,即num = 0,然后经过准备到初始化步骤,才把num真正的值5赋值给num,此时num = 5。
2、类加载器
类加载器负责加载所有的类,并为被加载的类生成Class对象。一旦一个类被加载到JVM中,同一个类就不会再次被加载,正如一个对象有唯一的标识一样,一个加载到JVM中类有一个唯一标识,在Java中,一个类用其全限定类名作为标识;但是在JVM中,一个类用其全限定类名和其类的类加载器作为唯一标识。JVM预定义三种类加载器,当一个JVM启动的时候,Java开始使用这三种类加载器加载class文件。
2.1、根类加载器
它是用来加载Java的核心类,使用原生代码来实现的,并不继承自ClassLoader,负责加载JAVA_HOME中jre/lib/*.jar,由C++实现。由于引导类加载器涉及到虚拟机本地实现细节,开发者无法直接获取到启动类加载器的引用,所以不允许直接用过引用进行操作。
2.2、扩展类加载器
它负责加载jre的扩展目录,加载lib/ext/*jar或者由java.ext.dirs系统属性指定的目录中的jar。由Java语言实现,父类加载器为null。
2.3、系统类加载器
也被称为应用类加载器,它负责在JVM启动时加载来自Java命令的-classpath选项、java.class.path系统属性、CLASSPATH换成变量所指定的jar包和类路径。程序可通过ClassLoader的静态方法getSystemClassLoader()来获取系统类加载器,由Java语言实现,父类加载器为扩展类加载器。
3、类的加载机制
3.1、JVM类加载三种机制
-
双亲委派机制:所谓的双亲委派机制,是先让父类加载器试图加载该class,只有在父类加载器无法加载该类时才会尝试从自己的类路径中加载该类。
-
全盘负责机制:是当一个加载器负责加载某个class时,该class所依赖和引用其他class也将由该类加载器负责载入,除非显式使用另外一个类加载器来载入。
-
缓存机制:缓存机制将会保证所有加载过的class都会被缓存,当程序中需要使用某个class时,类加载器先从缓存区中搜寻该class,只有当缓存区中不存在该class对象时,系统才会读取该类对应的二进制数据,并将其转换成class对象,存入缓冲区中。这也是为什么修改class后,需要重启JVM,程序所做修改才会生效的原因。
4、类加载器加载类的步骤
- 1、检测此class是否被加载过,即在缓存中是否有此class,如果有直接进入第8步,否则进入第2步。
- 2、如果没有父类加载器,可能父类是根类加载器(扩展类加载器没有父类加载器,因为它上层加载器是C++语言开发的根类加载器),也可能本身就是根类加载器,则进入第4步,如果有父类加载器,则进入到第3步。
- 3、请求使用父类加载器去加载目标类,如果成功加载则进入第8步,否则进入第5步。
- 4、请求使用根类加载器加载目标类,如果加载成功则进入第8步,失败则进入第7步。
- 5、当前类加载器尝试寻找目标类,如果找到则进入第6步,如果找不到则进入第7步。
- 6、从目标文件加载class,成功后进入第8步。
- 7、抛出ClassNotFoundException异常。
- 8、返回对应的Class对象。