在java中类的加载、初始化都是在程序运行期完成的,虽然会稍微增加开销,但是却很大的增加了灵活性,我们可用在运行期间动态的去网络或其他地方加载一个二进制流来作为程序代码的一部分。接下来我们简单介绍下java类加载过程。
从上图中我们可以看到类的生命周期主要有7个过程,分别为加载、验证、准备、解析、初始化、使用、卸载。
- 加载:
这个过程很简单,就是通过ClassLoader去加载Class文件到内存中,当然Class文件的来源并没有任何限制,既可以是我们的程序通过javac编译器生成的,也可以来源于网络等其他途径。 - 验证:
这个过程主要就是对Class文件中的字节流进行检查,看看是否符合规范,是否存在安全问题等等。 - 准备:
这个过程主要是为类变量(static修饰的变量)分配内存并设置初始值,此时的初始值指的是通常情况下的零值,例如private static int a = 3,类变量a在这个阶段会被赋值为0。唯一例外的类变量是使用final修饰的类变量,使用final修饰的类变量会在这个阶段就赋给程序中设定的值。 - 解析:
这个阶段主要将Class文件中对常量池的引用转为直接内存引用,举个简单的例子:我们常会在代码中使用import xxx.xxx.xxx.class,这里面的xxx.xxx.xxx.class就是符号引用,解析阶段就是把这种符号引用转换为对xxx.class具体内存地址的引用。 - 初始化:
这个阶段主要负责初始化类对象(Class对象),会按照在文件中出现的顺序为所有类变量赋值,同时执行静态代码块。千万注意这里初始化的是类对象而不是实例对象! - 使用、卸载:
这两个阶段就没啥可介绍的了,就是我们程序使用对象的过程,和对象生命周期结束后进行类型卸载的过程。
最后总结下对象静态属性/实例属性初始化的顺序:
(1)父类和子类的final static属性初始化
(2)父类的static属性初始化、父类的static代码块 (两者按在文件中的声明顺序先后初始化)
(3)子类的static属性初始化、子类的static代码块 (两者按在文件中的声明顺序先后初始化)
(4)父类的非静态属性、父类的非静态代码块 (两者按在文件中的声明顺序先后初始化)
(5)父类的构造函数执行
(6)子类的非静态属性、 子类的非静态代码块 (两者按在文件中的声明顺序先后初始化)
(7)子类的构造函数执行