11<!-- TOC -->
22
33- [ 类的生命周期] ( #类的生命周期 )
4- - [ 类加载过程] ( #类加载过程 )
5- - [ 加载] ( #加载 )
6- - [ 验证] ( #验证 )
7- - [ 准备] ( #准备 )
8- - [ 解析] ( #解析 )
9- - [ 初始化] ( #初始化 )
10- - [ 卸载] ( #卸载 )
11- - [ 公众号] ( #公众号 )
4+ - [ 类加载过程] ( #类加载过程 )
5+ - [ 加载] ( #加载 )
6+ - [ 验证] ( #验证 )
7+ - [ 准备] ( #准备 )
8+ - [ 解析] ( #解析 )
9+ - [ 初始化] ( #初始化 )
10+ - [ 卸载] ( #卸载 )
11+ - [ 公众号] ( #公众号 )
1212
1313<!-- /TOC -->
1414
1515# 类的生命周期
1616
1717一个类的完整生命周期如下:
1818
19-
2019![ ] ( https://my-blog-to-use.oss-cn-beijing.aliyuncs.com/2019-11/类加载过程-完善.png )
2120
2221## 类加载过程
@@ -29,13 +28,13 @@ Class 文件需要加载到虚拟机中之后才能运行和使用,那么虚
2928
3029### 加载
3130
32- 类加载过程的第一步,主要完成下面3件事情 :
31+ 类加载过程的第一步,主要完成下面 3 件事情 :
3332
34331 . 通过全类名获取定义此类的二进制字节流
35342 . 将字节流所代表的静态存储结构转换为方法区的运行时数据结构
36- 3 . 在内存中生成一个代表该类的 Class 对象,作为方法区这些数据的访问入口
35+ 3 . 在内存中生成一个代表该类的 ` Class ` 对象,作为方法区这些数据的访问入口
3736
38- 虚拟机规范上面这3点并不具体 ,因此是非常灵活的。比如:"通过全类名获取定义此类的二进制字节流" 并没有指明具体从哪里获取、怎样获取。比如:比较常见的就是从 ZIP 包中读取(日后出现的JAR、 EAR、WAR格式的基础 )、其他文件生成(典型应用就是JSP )等等。
37+ 虚拟机规范上面这 3 点并不具体 ,因此是非常灵活的。比如:"通过全类名获取定义此类的二进制字节流" 并没有指明具体从哪里获取、怎样获取。比如:比较常见的就是从 ` ZIP ` 包中读取(日后出现的 ` JAR ` 、 ` EAR ` 、 ` WAR ` 格式的基础 )、其他文件生成(典型应用就是 ` JSP ` )等等。
3938
4039** 一个非数组类的加载阶段(加载阶段获取类的二进制字节流的动作)是可控性最强的阶段,这一步我们可以去完成还可以自定义类加载器去控制字节流的获取方式(重写一个类加载器的 ` loadClass() ` 方法)。数组类型不通过类加载器创建,它由 Java 虚拟机直接创建。**
4140
@@ -51,16 +50,19 @@ Class 文件需要加载到虚拟机中之后才能运行和使用,那么虚
5150
5251** 准备阶段是正式为类变量分配内存并设置类变量初始值的阶段** ,这些内存都将在方法区中分配。对于该阶段有以下几点需要注意:
5352
54- 1 . 这时候进行内存分配的仅包括类变量(static),而不包括实例变量,实例变量会在对象实例化时随着对象一块分配在 Java 堆中。
55- 2 . 这里所设置的初始值"通常情况"下是数据类型默认的零值(如0、0L、null、false等),比如我们定义了` public static int value=111 ` ,那么 value 变量在准备阶段的初始值就是 0 而不是111(初始化阶段才会赋值)。特殊情况:比如给 value 变量加上了 final 关键字` public static final int value=111 ` ,那么准备阶段 value 的值就被赋值为 111。
53+ 1 . 这时候进行内存分配的仅包括类变量( Class Variables ,即静态变量,被 ` static ` 关键字修饰的变量,只与类相关,因此被称为类变量),而不包括实例变量。实例变量会在对象实例化时随着对象一块分配在 Java 堆中。
54+ 2 . 从概念上讲,类变量所使用的内存都应当在 ** 方法区** 中进行分配。不过有一点需要注意的是:JDK 7 之前,HotSpot 使用永久代来实现方法区的时候,实现是完全符合这种逻辑概念的。 而在 JDK 7 及之后,HotSpot 已经把原本放在永久代的字符串常量池、静态变量等移动到堆中,这个时候类变量则会随着 Class 对象一起存放在 Java 堆中。相关阅读:[ 《深入理解Java虚拟机(第3版)》勘误#75 ] ( https://github.com/fenixsoft/jvm_book/issues/75 )
55+ 3 . 这里所设置的初始值"通常情况"下是数据类型默认的零值(如 0、0L、null、false 等),比如我们定义了` public static int value=111 ` ,那么 value 变量在准备阶段的初始值就是 0 而不是 111(初始化阶段才会赋值)。特殊情况:比如给 value 变量加上了 final 关键字` public static final int value=111 ` ,那么准备阶段 value 的值就被赋值为 111。
56+
57+ ** 基本数据类型的零值** : (图片来自《深入理解 Java 虚拟机》第 3 版 7.33 )
58+
5659
57- ** 基本数据类型的零值:**
5860
5961![ 基本数据类型的零值] ( https://my-blog-to-use.oss-cn-beijing.aliyuncs.com/2019-6/基本数据类型的零值.png )
6062
6163### 解析
6264
63- 解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程。解析动作主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄和调用限定符7类符号引用进行 。
65+ 解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程。解析动作主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄和调用限定符 7 类符号引用进行 。
6466
6567符号引用就是一组符号来描述目标,可以是任何字面量。** 直接引用** 就是直接指向目标的指针、相对偏移量或一个间接定位到目标的句柄。在程序实际运行时,只有符号引用是不够的,举个例子:在程序执行方法时,系统需要明确知道这个方法所在的位置。Java 虚拟机为每个类都准备了一张方法表来存放类中所有的方法。当需要调用一个类的方法的时候,只要知道这个方法在方法表中的偏移量就可以直接调用该方法了。通过解析操作符号引用就可以直接转变为目标方法在类中方法表的位置,从而使得方法可以被调用。
6668
@@ -74,49 +76,48 @@ Class 文件需要加载到虚拟机中之后才能运行和使用,那么虚
7476
7577对于` <clinit> () ` 方法的调用,虚拟机会自己确保其在多线程环境中的安全性。因为 ` <clinit> () ` 方法是带锁线程安全,所以在多线程环境下进行类初始化的话可能会引起多个进程阻塞,并且这种阻塞很难被发现。
7678
77- 对于初始化阶段,虚拟机严格规范了有且只有5种情况下 ,必须对类进行初始化(只有主动去使用类才会初始化类):
79+ 对于初始化阶段,虚拟机严格规范了有且只有 5 种情况下 ,必须对类进行初始化(只有主动去使用类才会初始化类):
7880
79- 1 . 当遇到 new 、 getstatic、putstatic或invokestatic 这4条直接码指令时 ,比如 new 一个类,读取一个静态字段(未被 final 修饰)、或调用一个类的静态方法时。
80- - 当jvm执行new指令时会初始化类 。即当程序创建一个类的实例对象。
81- - 当jvm执行getstatic指令时会初始化类 。即程序访问类的静态变量(不是静态常量,常量会被加载到运行时常量池)。
82- - 当jvm执行putstatic指令时会初始化类 。即程序给类的静态变量赋值。
83- - 当jvm执行invokestatic指令时会初始化类 。即程序调用类的静态方法。
84- 2 . 使用 ` java.lang.reflect ` 包的方法对类进行反射调用时如Class .forname("..."), newInstance()等等。如果类没初始化,需要触发其初始化。
81+ 1 . 当遇到 ` new ` 、 ` getstatic ` 、 ` putstatic ` 或 ` invokestatic ` 这 4 条直接码指令时 ,比如 ` new ` 一个类,读取一个静态字段(未被 final 修饰)、或调用一个类的静态方法时。
82+ - 当 jvm 执行 new 指令时会初始化类 。即当程序创建一个类的实例对象。
83+ - 当 jvm 执行 getstatic 指令时会初始化类 。即程序访问类的静态变量(不是静态常量,常量会被加载到运行时常量池)。
84+ - 当 jvm 执行 putstatic 指令时会初始化类 。即程序给类的静态变量赋值。
85+ - 当 jvm 执行 invokestatic 指令时会初始化类 。即程序调用类的静态方法。
86+ 2 . 使用 ` java.lang.reflect ` 包的方法对类进行反射调用时如 Class .forname("..."), newInstance()等等。如果类没初始化,需要触发其初始化。
85873 . 初始化一个类,如果其父类还未初始化,则先触发该父类的初始化。
86884 . 当虚拟机启动时,用户需要定义一个要执行的主类 (包含 main 方法的那个类),虚拟机会先初始化这个类。
87- 5 . MethodHandle和VarHandle可以看作是轻量级的反射调用机制,而要想使用这2个调用 ,
88- 就必须先使用findStaticVarHandle来初始化要调用的类 。
89- 6 . ** 「补充,来自[ issue745] ( https://github.com/Snailclimb/JavaGuide/issues/745 ) 」** 当一个接口中定义了JDK8新加入的默认方法(被default关键字修饰的接口方法 )时,如果有这个接口的实现类发生了初始化,那该接口要在其之前被初始化。
89+ 5 . MethodHandle 和 VarHandle 可以看作是轻量级的反射调用机制,而要想使用这 2 个调用 ,
90+ 就必须先使用 findStaticVarHandle 来初始化要调用的类 。
91+ 6 . ** 「补充,来自[ issue745] ( https://github.com/Snailclimb/JavaGuide/issues/745 ) 」** 当一个接口中定义了 JDK8 新加入的默认方法(被 default 关键字修饰的接口方法 )时,如果有这个接口的实现类发生了初始化,那该接口要在其之前被初始化。
9092
9193## 卸载
9294
93- > 卸载这部分内容来自 [ issue #662 ] ( https://github.com/Snailclimb/JavaGuide/issues/662 ) 由 ** [ guang19] ( https://github.com/guang19 ) ** 补充完善。
95+ > 卸载这部分内容来自 [ issue #662 ] ( https://github.com/Snailclimb/JavaGuide/issues/662 ) 由 ** [ guang19] ( https://github.com/guang19 ) ** 补充完善。
9496
95- 卸载类即该类的Class对象被GC 。
97+ 卸载类即该类的 Class 对象被 GC 。
9698
97- 卸载类需要满足3个要求 :
99+ 卸载类需要满足 3 个要求 :
98100
99- 1 . 该类的所有的实例对象都已被GC ,也就是说堆不存在该类的实例对象。
101+ 1 . 该类的所有的实例对象都已被 GC ,也就是说堆不存在该类的实例对象。
1001022 . 该类没有在其他任何地方被引用
101- 3 . 该类的类加载器的实例已被GC
103+ 3 . 该类的类加载器的实例已被 GC
102104
103- 所以,在JVM生命周期类,由jvm自带的类加载器加载的类是不会被卸载的 。但是由我们自定义的类加载器加载的类是可能被卸载的。
105+ 所以,在 JVM 生命周期类,由 jvm 自带的类加载器加载的类是不会被卸载的 。但是由我们自定义的类加载器加载的类是可能被卸载的。
104106
105- 只要想通一点就好了,jdk自带的BootstrapClassLoader, ExtClassLoader, AppClassLoader负责加载jdk提供的类 ,所以它们(类加载器的实例)肯定不会被回收。而我们自定义的类加载器的实例是可以被回收的,所以使用我们自定义加载器加载的类是可以被卸载掉的。
107+ 只要想通一点就好了,jdk 自带的 ` BootstrapClassLoader ` , ` ExtClassLoader ` , ` AppClassLoader ` 负责加载 jdk 提供的类 ,所以它们(类加载器的实例)肯定不会被回收。而我们自定义的类加载器的实例是可以被回收的,所以使用我们自定义加载器加载的类是可以被卸载掉的。
106108
107109** 参考**
108110
109- - 《深入理解Java虚拟机 》
110- - 《实战Java虚拟机 》
111+ - 《深入理解 Java 虚拟机 》
112+ - 《实战 Java 虚拟机 》
111113- < https://docs.oracle.com/javase/specs/jvms/se7/html/jvms-5.html >
112114
113115## 公众号
114116
115117如果大家想要实时关注我更新的文章以及分享的干货的话,可以关注我的公众号。
116118
117- ** 《Java面试突击》:** 由本文档衍生的专为面试而生的《Java面试突击》V2.0 PDF 版本[ 公众号] ( #公众号 ) 后台回复 ** "Java面试突击"** 即可免费领取!
118-
119- ** Java工程师必备学习资源:** 一些Java工程师常用学习资源[ 公众号] ( #公众号 ) 后台回复关键字 ** “1”** 即可免费无套路获取。
119+ ** 《Java 面试突击》:** 由本文档衍生的专为面试而生的《Java 面试突击》V2.0 PDF 版本[ 公众号] ( #公众号 ) 后台回复 ** "Java 面试突击"** 即可免费领取!
120120
121- ![ 我的公众号 ] ( https://my-blog-to-use.oss-cn-beijing.aliyuncs.com/2019-6/167598cd2e17b8ec.png )
121+ ** Java 工程师必备学习资源: ** 一些 Java 工程师常用学习资源 [ 公众号 ] ( #公众号 ) 后台回复关键字 ** “1” ** 即可免费无套路获取。
122122
123+ ![ 我的公众号] ( https://my-blog-to-use.oss-cn-beijing.aliyuncs.com/2019-6/167598cd2e17b8ec.png )
0 commit comments