类的生命周期
一个类的完整生命周期如下:
类加载过程
class文件需要加载到虚拟机中之后才能运行和使用,那么虚拟机是如何加载这些class文件的呢?
系统加载class类型的文件主要三步:加载->连接->初始化.连接过程又可以分为三步:验证->准备->解析
加载
类加载过程的第一步,主要完成下面三件事情:
- 通过全类名获取定义此类的二进制字节流
- 将字节流所代表的静态存储结构转换为方法区的运行时数据结构
- 在内存中生成一个代表该类的class对象,作为方法区这些数据的访问入口
虚拟机规范上这三点并不具体,因此是非常灵活的,比如通过全类名获取定义此类的二进制字节流。并没有指明具体从哪里获取,怎样获取。比如比较常见的就是从zip包中读取(日后出现的jar,ear,war格式的基础),其他文件生成等等。
一个非数组类的加载阶段(加载阶段获取类的二进制字节流的动作)是可控性最强的阶段,这一步我们可以去完成还可以自定义类加载器去控制字节流的获取方式(重写一个类加载器的loadClass()方法),数组类型不通过类加载器创建,它由java虚拟机直接创建。
类加载器,双亲委派模型也是非常重要的知识点。这部分内容会在后面的笔记中单独介绍。
加载阶段和连接阶段的部分内容是交叉进行的,加载阶段尚未结束,连接阶段可能就已经开始了。
验证
准备
准备阶段是正式为类变量分配内存并设置类变量初始值的阶段。这些内存都将在方法区中分配,对于该阶段有以下几点需要注意:
- 这时候进行内存分配的仅仅包括类变量(即静态变量,被static关键字修饰的变量,只与类相关,也被称为类变量).而不包括实例变量,实例变量会在对象实例化时随着对象一块分配在java堆中。
- 从概念上讲,类变量所使用的内存都应当在方法区中进行分配, 不过有一点需要注意的是:jdk7之前,hotspot使用永久代来实现方法区的时候,实现是完全符合这一逻辑概念的。而在jdk7之后,hotspot已经把原本放在永久代的字符串常量池,静态变量移动到堆中,这个时候类变量则会随着class对象一起存放在java堆中。
- 这里所设置的初始值通常情况下是数据类型默认的零值。如0,0l,null,false等,比如我们定义了public static int value = 111.那么value变量在准备阶段的初始值是0而不是111,初始化阶段才会赋值。特殊情况:比如给value变量加上了final关键字,那么在准备阶段value的值就会被赋值为111.
解析
解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程,解析动作主要针对类或接口,字段,类方法,接口方法,方法类型,方法句柄和调用限定符七类符号引用进行。
符号引用就是一组符号来描述目标,可以是任何字面量。直接引用就是直接指向目标的指针,相对偏移量或一个间接定位到目标的句柄。在程序实际运行时,只有符号引用是不够的,举个例子:在程序执行方法时,系统需要明确知道这个方法所在的位置,java虚拟机为每个类都准备了一张方法表来存放类中的所有方法。当需要调用一个类的方法的时候,只要知道这个方法在方法表中的偏移量就可以直接调用该方法了。通过解析操作符号引用就可以直接转变为目标方法在类中方法表的位置,从而使得方法可以被调用。
综上,解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程,也就是得到类或者字段,方法在内存中的指针或者偏移量。
初始化
初始化阶段是执行初始化方法<clinit>()方法的过程,是类加载的最后一步。这一步jvm才开始真正执行类中定义的java程序代码。
<clinit>()方法是编译后自动生成的
对于<clinit>()方法的调用,虚拟机会自己确保其在多线程环境中的安全性。因为<clinit>()方法是带锁线程安全,所以在多线程环境下进行类初始化的话可能会引起多个进程阻塞,并且这种阻塞很难被发现。
对于初始化阶段,虚拟机严格规范了有且只有6种情况下,必须对类进行初始化(只有主动去使用类才会初始化类):
- 当遇到new,getstatic,putstatic或者invokestatic这四条字节码指令时,比如new一个类,读取一个静态字段或者调用类的静态方法时。
- 当jvm执行new指令时会初始化类,即当程序创建一个类的实例对象。
- 当jvm执行getstatic指令时会初始化类。即程序访问类的静态变量(不是静态变量,常量会被加载到运行时常量池)。
- 当jvm执行putstatic指令时会初始化类,即程序给类的静态变量赋值。
- 当jvm执行invokestatic指令时会初始化类,即程序调用类的静态方法。
- 使用java.lang.reflect包的方法对类进行反射调用时。如Class.forname("...").newInstance()等等,如果类没有初始化,需要触发其初始化。
- 初始化一个类,如果其父类还未初始化,则先触发该父类的初始化。
- 当虚拟机启动时,用户需要定义一个要执行的主类(包含main方法的那个类),虚拟机就会先初始化这个类。。
- MethodHandle和VarHandle可以看作是轻量级的反射调用机制,而想要使用这两个调用,就必须先使用findStaticVarHandle来初始化要调用的类。
- 当一个接口中定义了JDK8新加入的默认方法(被default关键字修饰的接口方法)时,如果有这个接口的实现类发生了初始化,那么该接口要在其之前被初始化。
卸载
卸载类即该类的class对象被GC。
卸载类需要满足3个要求:
- 该类的所有的实例对象都已被GC,也就是说堆不存在该类的实例对象。
- 该类没有在其他任何地方被引用
- 该类的类加载器的实例已被GC
所以在JVM生命周期内,由jvm自带的类加载器加载的类是不会被卸载的。但是由我们自定义的类加载器加载的类是可能被卸载的。
本篇笔记就记到这里,如果稍微帮到你了记得点个喜欢点个关注。也祝大家工作顺顺利利,生活愉快~!