类加载过程:
-
加载
- 通过类的全限定名加载类的二进制字节流
- 将字节流所代表的静态存储结构转化为方法区的运行时数据结构
- 在内存生成一个代表这个类的Class对象,用来做访问方法区对应这个类的各种数据的入口
-
验证
为了确保加载进来的Class文件的字节流符合虚拟机的要求,防止威胁虚拟机的安全
-
准备
为类变量分配内存并设置初始值,因为是类变量,所以这些变量都会在方法区分配,要注意的是初始值是指默认初始值,比如如果是int,在这个阶段会被设置为0
-
解析
(这个阶段是我最理解不了的阶段)是将常量池的符合引用替换成直接引用的过程(妈个鸡这是什么意思?)
-
初始化
真正的为静态变量赋值以及正式执行静态代码块,这个阶段有一些点要注意:
-
静态代码块不能使用在它后面定义的变量,只能赋值不能访;
public class Test { static { i = 0; // 给变量赋值可以正常编译通过 System.out.print(i); // 这句编译器回提示“非法向前引用” } static int i = 1; }
-
虚拟机会保证父类的初始化先于子类,所以静态代码执行以及静态变量的赋值一定是父类先开始的
static class Parent { public static int A = 1; static { A = 2 } } static class Sub extends Parent { public static int B = A; } public static void main(String[] args) { System.out.println(Sub.A); // 2 System.out.println(Sub.B); // 1 } /* 顺序为: int A = 1; static { A = 2 } int B = A; 所以最终B的指是2 */
这里基于以上顺嘴提一个问题,loadClass和Class.forName的区别:
loadClass不会执行初始化阶段,所以静态代码不会执行以及静态变量不会赋值
而Class.forName会执行初始化阶段,所以会执行静态代码以及会对静态变量赋值
因为jdbc需要执行静态代码块来注册驱动,所以用了Class.forName
-