当new一个类对象时,如果该类还未被JVM所加载,则JVM首先加载该类,加载类流程中会有类的初始化操作,也就是类中static域的初始化,即调用<cinit>方法;在创建对象时,会进行对象变量的初始化操作,即调用<init>方法;最后调用对象的构造方法。
在类中static域初始化流程中,是按照前后顺序来进行的,也就是不管是static块还是static变量,谁在前面谁就先初始化;在对象变量的初始化操作中,也是按照前后顺序来进行的,也就是不管是构造代码块还是类属性变量,谁在前面谁就先初始化。
类的静态成员 --> 类的实例成员 --> 类的构造方法
说到这里,有一个很好的问题。比如一个类中静态变量是一个该类的实例,那么此时类的初始化顺序是什么样的呢?
public class Alibaba {
public static int k = 0;
public static Alibaba t1 = new Alibaba("t1");
public static Alibaba t2 = new Alibaba("t2");
public static int i = print("i");
public static int n = 99;
private int a = 0;
public int j = print("j");
{
print("构造块");
}
static {
print("静态块");
}
public Alibaba(String str) {
System.out.println((++k) + ":" + str + " i=" + i + " n=" + n);
++i;
++n;
}
public static int print(String str) {
System.out.println((++k) + ":" + str + " i=" + i + " n=" + n);
++n;
return ++i;
}
public static void main(String args[]) {
Alibaba t = new Alibaba("init");
}
}
据说上述代码是阿里14年校招题,输出结果为:
1:j i=0 n=0
2:构造块 i=1 n=1
3:t1 i=2 n=2
4:j i=3 n=3
5:构造块 i=4 n=4
6:t2 i=5 n=5
7:i i=6 n=6
8:静态块 i=7 n=99
9:j i=8 n=100
10:构造块 i=9 n=101
11:init i=10 n=102
上面就是输出结果,根据输出结果,我们来分析下:
(1) 由于Alibaba类是JVM启动类,所以首先被加载,也就是执行加载、连接、初始化这些流程。
(2) 进过加载和连接流程后,进入初始化阶段,该阶段会对static域进行初始化。
(3) 首先k被赋值为0,接下来是t1,由于这时Alibaba类已经处于初始化阶段了,static无需再次初始化,否则会导致static域多次初始化情况,所以暂时忽略static域初始化操作,对非static域进行初始化操作,于是就有了开始的:
1:j i=0 n=0
2:构造块 i=1 n=1
3:t1 i=2 n=2
(4) 接着对t2赋值,过程与t1一样:
4:j i=3 n=3
5:构造块 i=4 n=4
6:t2 i=5 n=5
(5) 然后对static变量i和n赋值,还有静态块初始化:
7:i i=6 n=6
8:静态块 i=7 n=99
(6) 此时已完成static域的初始化工作。执行new Alibaba("init")
时,会触发对象初始化操作,也就是以下输出了:
9:j i=8 n=100
10:构造块 i=9 n=101
11:init i=10 n=102
参考:
1、http://liujiacai.net/blog/2014/07/12/order-of-initialization-in-java/
网友评论