jvm类加载详解

在这里插入图片描述

引言

如下图所示,JVM类加载机制分为五个部分:加载,验证,准备,解析,初始化,下面我们就分别来看一下这五个过程。

JVM系列文章

JVM内存模型解析
JVM垃圾回收算法及回收器详解
加载

加载是类加载过程中的一个阶段,这个阶段会在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据的入口。注意这里不一定非得要从一个Class文件获取,这里既可以从ZIP包中读取(比如从jar包和war包中读取),也可以在运行时计算生成(动态代理),也可以由其它文件生成(比如将JSP文件转换成对应的Class类)。

验证

这一阶段的主要目的是为了确保Class文件的字节流中包含的信息是否符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。

准备

准备阶段是正式为类变量分配内存并设置类变量的初始值阶段,即在方法区中分配这些变量所使用的内存空间。注意这里所说的初始值概念,比如一个类变量定义为:

public static int v = 8080;
实际上变量v在准备阶段过后的初始值为0而不是8080,将v赋值为8080的putstatic指令是程序被编译后,存放于类构造器方法之中,这里我们后面会解释。

但是注意如果声明为:

public static final int v = 8080;
在编译阶段会为v生成ConstantValue属性,在准备阶段虚拟机会根据ConstantValue属性将v赋值为8080。

解析

解析阶段是指虚拟机将常量池中的符号引用替换为直接引用的过程。符号引用就是class文件中的:

CONSTANT_Class_info
CONSTANT_Field_info
CONSTANT_Method_info
等类型的常量。

下面我们解释一下符号引用和直接引用的概念:

符号引用与虚拟机实现的布局无关,引用的目标并不一定要已经加载到内存中。各种虚拟机实现的内存布局可以各不相同,但是它们能接受的符号引用必须是一致的,因为符号引用的字面量形式明确定义在Java虚拟机规范的Class文件格式中。

直接引用可以是指向目标的指针,相对偏移量或是一个能间接定位到目标的句柄。如果有了直接引用,那引用的目标必定已经在内存中存在。

初始化

初始化阶段是类加载最后一个阶段,前面的类加载阶段之后,除了在加载阶段可以自定义类加载器以外,其它操作都由JVM主导。到了初始阶段,才开始真正执行类中定义的Java程序代码。

初始化阶段是执行类构造器方法的过程。方法是由编译器自动收集类中的类变量的赋值操作和静态语句块中的语句合并而成的。虚拟机会保证方法执行之前,父类的方法已经执行完毕。p.s: 如果一个类中没有对静态变量赋值也没有静态语句块,那么编译器可以不为这个类生成()方法。

注意以下几种情况不会执行类初始化:

通过子类引用父类的静态字段,只会触发父类的初始化,而不会触发子类的初始化。
定义对象数组,不会触发该类的初始化。
常量在编译期间会存入调用类的常量池中,本质上并没有直接引用定义常量的类,不会触发定义常量所在的类。
通过类名获取Class对象,不会触发类的初始化。
通过Class.forName加载指定类时,如果指定参数initialize为false时,也不会触发类初始化,其实这个参数是告诉虚拟机,是否要对类进行初始化。
通过ClassLoader默认的loadClass方法,也不会触发初始化动作。
类加载器

虚拟机设计团队把加载动作放到JVM外部实现,以便让应用程序决定如何获取所需的类,JVM提供了3种类加载器:

启动类加载器(Bootstrap ClassLoader):负责加载 JAVA_HOME\lib 目录中的,或通过-Xbootclasspath参数指定路径中的,且被虚拟机认可(按文件名识别,如rt.jar)的类。
扩展类加载器(Extension ClassLoader):负责加载 JAVA_HOME\lib\ext 目录中的,或通过java.ext.dirs系统变量指定路径中的类库。
应用程序类加载器(Application ClassLoader):负责加载用户路径(classpath)上的类库。
JVM通过双亲委派模型进行类的加载,当然我们也可以通过继承java.lang.ClassLoader实现自定义的类加载器。

jvm classloaderjvm classloader

当一个类加载器收到类加载任务,会先交给其父类加载器去完成,因此最终加载任务都会传递到顶层的启动类加载器,只有当父类加载器无法完成加载任务时,才会尝试执行加载任务。

采用双亲委派的一个好处是比如加载位于rt.jar包中的类java.lang.Object,不管是哪个加载器加载这个类,最终都是委托给顶层的启动类加载器进行加载,这样就保证了使用不同的类加载器最终得到的都是同样一个Object对象。

在有些情境中可能会出现要我们自己来实现一个类加载器的需求,由于这里涉及的内容比较广泛,我想以后单独写一篇文章来讲述,不过这里我们还是稍微来看一下。我们直接看一下jdk中的ClassLoader的源码实现:

protected synchronized Class<?> loadClass(String name, boolean resolve)
throws ClassNotFoundException {
// First, check if the class has already been loaded
Class c = findLoadedClass(name);
if (c == null) {
try {
if (parent != null) {
c = parent.loadClass(name, false);
} else {
c = findBootstrapClass0(name);
}
} catch (ClassNotFoundException e) {
// If still not found, then invoke findClass in order
// to find the class.
c = findClass(name);
}
}
if (resolve) {
resolveClass©;
}
return c;
}
首先通过Class c = findLoadedClass(name);判断一个类是否已经被加载过。
如果没有被加载过执行if (c == null)中的程序,遵循双亲委派的模型,首先会通过递归从父加载器开始找,直到父类加载器是Bootstrap ClassLoader为止。
最后根据resolve的值,判断这个class是否需要解析。
而上面的findClass()的实现如下,直接抛出一个异常,并且方法是protected,很明显这是留给我们开发者自己去实现的,这里我们以后我们单独写一篇文章来讲一下如何重写findClass方法来实现我们自己的类加载器。

protected Class<?> findClass(String name) throws ClassNotFoundException {
throw new ClassNotFoundException(name);
}

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
pbV5sA4L9U7QL/GLh+gMZQ==

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值