类加载过程.md 7.9 KB
Newer Older
S
SnailClimb 已提交
1
<!-- TOC -->
S
SnailClimb 已提交
2

S
shuang.kou 已提交
3 4 5 6 7 8 9 10 11
- [类的生命周期](#类的生命周期)
    - [类加载过程](#类加载过程)
        - [加载](#加载)
        - [验证](#验证)
        - [准备](#准备)
        - [解析](#解析)
        - [初始化](#初始化)
    - [卸载](#卸载)
    - [公众号](#公众号)
S
SnailClimb 已提交
12

S
SnailClimb 已提交
13
<!-- /TOC -->
S
SnailClimb 已提交
14

S
shuang.kou 已提交
15
# 类的生命周期
S
SnailClimb 已提交
16

S
shuang.kou 已提交
17 18 19 20 21 22
一个类的完整生命周期如下:


![](https://my-blog-to-use.oss-cn-beijing.aliyuncs.com/2019-11/类加载过程-完善.png)

## 类加载过程
S
SnailClimb 已提交
23

S
SnailClimb 已提交
24
Class 文件需要加载到虚拟机中之后才能运行和使用,那么虚拟机是如何加载这些 Class 文件呢?
S
SnailClimb 已提交
25

S
SnailClimb 已提交
26
系统加载 Class 类型的文件主要三步:**加载->连接->初始化**。连接过程又可分为三步:**验证->准备->解析**
S
SnailClimb 已提交
27

S
shuang.kou 已提交
28
![](https://my-blog-to-use.oss-cn-beijing.aliyuncs.com/2019-6/类加载过程.png)
S
SnailClimb 已提交
29

S
shuang.kou 已提交
30
### 加载
S
SnailClimb 已提交
31 32 33 34 35 36 37

类加载过程的第一步,主要完成下面3件事情:

1. 通过全类名获取定义此类的二进制字节流
2. 将字节流所代表的静态存储结构转换为方法区的运行时数据结构
3. 在内存中生成一个代表该类的 Class 对象,作为方法区这些数据的访问入口

H
hailong.sha 已提交
38
虚拟机规范上面这3点并不具体,因此是非常灵活的。比如:"通过全类名获取定义此类的二进制字节流" 并没有指明具体从哪里获取、怎样获取。比如:比较常见的就是从 ZIP 包中读取(日后出现的JAR、EAR、WAR格式的基础)、其他文件生成(典型应用就是JSP)等等。
S
SnailClimb 已提交
39 40 41 42 43 44 45

**一个非数组类的加载阶段(加载阶段获取类的二进制字节流的动作)是可控性最强的阶段,这一步我们可以去完成还可以自定义类加载器去控制字节流的获取方式(重写一个类加载器的 `loadClass()` 方法)。数组类型不通过类加载器创建,它由 Java 虚拟机直接创建。**

类加载器、双亲委派模型也是非常重要的知识点,这部分内容会在后面的文章中单独介绍到。

加载阶段和连接阶段的部分内容是交叉进行的,加载阶段尚未结束,连接阶段可能就已经开始了。

S
shuang.kou 已提交
46
### 验证
S
SnailClimb 已提交
47

S
SnailClimb 已提交
48
![验证阶段示意图](https://my-blog-to-use.oss-cn-beijing.aliyuncs.com/2019-6/验证阶段.png)
S
SnailClimb 已提交
49

S
shuang.kou 已提交
50
### 准备
S
SnailClimb 已提交
51 52 53 54

**准备阶段是正式为类变量分配内存并设置类变量初始值的阶段**,这些内存都将在方法区中分配。对于该阶段有以下几点需要注意:

1. 这时候进行内存分配的仅包括类变量(static),而不包括实例变量,实例变量会在对象实例化时随着对象一块分配在 Java 堆中。
L
fix-571  
liwenguang 已提交
55
2. 这里所设置的初始值"通常情况"下是数据类型默认的零值(如0、0L、null、false等),比如我们定义了`public static int value=111` ,那么 value 变量在准备阶段的初始值就是 0 而不是111(初始化阶段才会赋值)。特殊情况:比如给 value 变量加上了 fianl 关键字`public static final int value=111` ,那么准备阶段 value 的值就被赋值为 111。
S
SnailClimb 已提交
56 57 58

**基本数据类型的零值:**

S
SnailClimb 已提交
59
![基本数据类型的零值](https://my-blog-to-use.oss-cn-beijing.aliyuncs.com/2019-6/基本数据类型的零值.png)
S
SnailClimb 已提交
60

S
shuang.kou 已提交
61
### 解析
S
SnailClimb 已提交
62 63 64

解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程。解析动作主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄和调用限定符7类符号引用进行。

葱花鸡蛋's avatar
葱花鸡蛋 已提交
65
符号引用就是一组符号来描述目标,可以是任何字面量。**直接引用**就是直接指向目标的指针、相对偏移量或一个间接定位到目标的句柄。在程序实际运行时,只有符号引用是不够的,举个例子:在程序执行方法时,系统需要明确知道这个方法所在的位置。Java 虚拟机为每个类都准备了一张方法表来存放类中所有的方法。当需要调用一个类的方法的时候,只要知道这个方法在方法表中的偏移量就可以直接调用该方法了。通过解析操作符号引用就可以直接转变为目标方法在类中方法表的位置,从而使得方法可以被调用。
S
SnailClimb 已提交
66 67 68

综上,解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程,也就是得到类或者字段、方法在内存中的指针或者偏移量。

S
shuang.kou 已提交
69
### 初始化
S
SnailClimb 已提交
70

H
fix  
hening1 已提交
71
初始化是类加载的最后一步,也是真正执行类中定义的 Java 程序代码(字节码),初始化阶段是执行初始化方法 `<clinit> ()`方法的过程。
S
SnailClimb 已提交
72 73 74

对于`<clinit>()` 方法的调用,虚拟机会自己确保其在多线程环境中的安全性。因为 `<clinit>()` 方法是带锁线程安全,所以在多线程环境下进行类初始化的话可能会引起死锁,并且这种死锁很难被发现。

S
shuang.kou 已提交
75
对于初始化阶段,虚拟机严格规范了有且只有5种情况下,必须对类进行初始化(只有主动去使用类才会初始化类):
S
SnailClimb 已提交
76 77

1. 当遇到 new 、 getstatic、putstatic或invokestatic 这4条直接码指令时,比如 new 一个类,读取一个静态字段(未被 final 修饰)、或调用一个类的静态方法时。
S
shuang.kou 已提交
78 79 80 81 82
   - 当jvm执行new指令时会初始化类。即当程序创建一个类的实例对象。
   - 当jvm执行getstatic指令时会初始化类。即程序访问类的静态变量(不是静态常量,常量会被加载到运行时常量池)。
   - 当jvm执行putstatic指令时会初始化类。即程序给类的静态变量赋值。
   - 当jvm执行invokestatic指令时会初始化类。即程序调用类的静态方法。
2. 使用 `java.lang.reflect` 包的方法对类进行反射调用时如Class.forname("..."),newInstance()等等。 ,如果类没初始化,需要触发其初始化。
S
SnailClimb 已提交
83 84
3. 初始化一个类,如果其父类还未初始化,则先触发该父类的初始化。
4. 当虚拟机启动时,用户需要定义一个要执行的主类 (包含 main 方法的那个类),虚拟机会先初始化这个类。
S
shuang.kou 已提交
85 86
5. MethodHandle和VarHandle可以看作是轻量级的反射调用机制,而要想使用这2个调用,
   就必须先使用findStaticVarHandle来初始化要调用的类。
S
shuang.kou 已提交
87
6. **「补充,来自[issue745](https://github.com/Snailclimb/JavaGuide/issues/745)」** 当一个接口中定义了JDK8新加入的默认方法(被default关键字修饰的接口方法)时,如果有这个接口的实现类发生了初始化,那该接口要在其之前被初始化。
S
SnailClimb 已提交
88

S
shuang.kou 已提交
89 90 91 92 93 94 95 96 97 98 99 100 101 102
## 卸载

> 卸载这部分内容来自 [issue#662](https://github.com/Snailclimb/JavaGuide/issues/662)由 **[guang19](https://github.com/guang19)**  补充完善。

卸载类即该类的Class对象被GC。

卸载类需要满足3个要求:

1. 该类的所有的实例对象都已被GC,也就是说堆不存在该类的实例对象。
2. 该类没有在其他任何地方被引用
3. 该类的类加载器的实例已被GC

所以,在JVM生命周期类,由jvm自带的类加载器加载的类是不会被卸载的。但是由我们自定义的类加载器加载的类是可能被卸载的。

H
hailong.sha 已提交
103
只要想通一点就好了,jdk自带的BootstrapClassLoader,ExtClassLoader,AppClassLoader负责加载jdk提供的类,所以它们(类加载器的实例)肯定不会被回收。而我们自定义的类加载器的实例是可以被回收的,所以使用我们自定义加载器加载的类是可以被卸载掉的。
S
shuang.kou 已提交
104

S
SnailClimb 已提交
105 106 107 108
**参考**

- 《深入理解Java虚拟机》
- 《实战Java虚拟机》
S
SnailClimb 已提交
109 110
- <https://docs.oracle.com/javase/specs/jvms/se7/html/jvms-5.html>

S
SnailClimb 已提交
111
## 公众号
S
SnailClimb 已提交
112 113 114 115 116 117 118

如果大家想要实时关注我更新的文章以及分享的干货的话,可以关注我的公众号。

**《Java面试突击》:** 由本文档衍生的专为面试而生的《Java面试突击》V2.0 PDF 版本[公众号](#公众号)后台回复 **"Java面试突击"** 即可免费领取!

**Java工程师必备学习资源:** 一些Java工程师常用学习资源[公众号](#公众号)后台回复关键字 **“1”** 即可免费无套路获取。 

S
Shuang Kou 已提交
119
![我的公众号](https://my-blog-to-use.oss-cn-beijing.aliyuncs.com/2019-6/167598cd2e17b8ec.png)
S
shuang.kou 已提交
120