一、jvm会在什么情况下加载一个类?
分析:什么是类加载?为什么会发生类加载?什么情况下发生类加载?
Java要运行,必须把源文件(.java)编译成字节码(.class),jvm的输入是字节码文件,需要把字节码文件加载到jvm内存中,要如何解析字节码文件,按照什么样的规则去解析,解析后放在内存中的什么位置,这些过程可以理解为类加载。一个java文件要运行必须通过类加载的过程,转化为JVM可以识别的数据结构。
为什么发生类加载,是由于要运行java程序,jvm不能识别java源程序,编译后的字节码文件jvm可以识别,但是要转化为jvm可以识别的数据结构,所以有类加载的过程。
java程序对类的使用分为主动使用和被动使用,只有主动使用才会发生类加载:
-
访问类的静态变量与静态方法,如果类没有加载,则首先会执行类加载
-
初始化一个类的子类的时候,如果子类没有加载,则首先会执行子类的加载
-
使用new关键字,生成一个对象,如果该类没有加载,则首先会加载该类
-
使用反射,Class.forName();如果该类没有加载,则执行该类的加载
-
标记为启动类的加载,如果没有加载则首先执行该类的加载,常见的是main方法所在的类
-
1.7提供的动态语言支持
二、jvm类加载的生命周期
类加载到使用会经历过程(依次进行):
加载->验证->准备->解析->初始化->使用->卸载
(1)加载
看下面这个代码:
public class DemoMain {
public static void main(String[] args) {
Demo demo = new Demo();
}
}
一旦JVM启动,就会把这个类加载到内存,然后从main方法的入口代码开始执行。
接着代码中实例化了Demo对象,此时就需要将"Demo.class"字节码文件的这个类加载到内存。
(2)验证
验证这一阶段是根据java虚拟机规范来校验加载进来的‘.class’文件中的内容,是否符合指定的规范(文件格式验证、元数据验证、字节码验证、符号引用验证等等)。如果你的‘.class’文件被篡改了,里面的字节码不符合规范,那么jvm就不能够执行。这样可以有效的提高安全性。
(3)准备
public class Demo {
public static int num;
}
假如有以上代码。准备工作其实就是给这个“Demo”类分配一定的内存空间;
然后给他里面的类变量(static修饰的变量)num分配内存空间,并且分配一个默认初始值“0”。
准备阶段较为核心,因为这个阶段会给加载进来的类与类变量分配内存空间,并给变量默认的初始值。
(4)解析
将常量池中的符号引用转化为直接引用的过程,包括类,字段,方法的解析;
(5)初始化
初始化阶段, 简言之, 为类的静态变量赋予正确的初始化值。到了初始化阶段, 才真正开始执行类中定义的java程序代码。
public class Demo {
public static int num = 5;
}
以上代码将5赋值给静态变量num,那么他会在准备阶段赋值吗?
当然不会,准备阶段仅仅是给number这个类变量开辟了一个内存空间,然后给个初始值“0”;
赋值代码是在“初始化”阶段来执行。
同样static静态代码块,也会在这个阶段来执行。
看看下面的代码:
public class ClassInitTest {
private static int number = 10;
static {
number = 20;
System.out.println(number);
}
public static void main(String[] args) {
System.out.println(ClassInitTest.number);
}
}
你觉得上述代码输出的最终结果是什么?以及number值的变化范围?
最终结果为20 ,变化范围是0-10-20
在虚拟机准备阶段分配内存空间,此时为number赋值为零,初始化的时候加载静态变量和静态代码块 (因为是在同一阶段加载),此时的顺序根据代码的顺序(从上到下),所以number先是赋值为10,最后在变为20.