前两天看到阿里笔试的附加题,第一道考到了对象的初始化顺序,题目如下:
public class Test {
public static int k = 0;
public static Test t1 = new Test("t1");
public static Test t2 = new Test("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 Test(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[]) {
Test t = new Test("init");
}
}
这道题考的太好了,初始化顺序这东西貌似都接触过,但真的没有太怎么注意过,借着这道题,我也复习下学过的一些东西。
在最开始接触Java的时候,Java的口号就是跨平台,所以我们都知道Java程序有这么个流程,就是.java--->.class--->可执行文件,我们可以看看在这些阶段都发生了什么,.java文件在被编译成.class文件的时候,主要进行的是对语法的检查,最后生成了JVM字节码文件,即class文件,这个过程相当于我们使用了javac xxx命令。在.class文件被加载到JVM中并运行的过程中,主要发生了三个过程,即加载、链接、初始化。简单的说,
加载:就是把class文件导入到JVM中的过程
链接:包括校验、准备、解析三个过程
校验:说白了就是检验class文件是不是正确的
准备:为类的静态变量分配空间,并赋初值
解析:把符号表示转换为直接表示
这个概念刚开始我也不太懂,参考了http://blog.csdn.net/imzoer/article/details/8086255这篇文章,讲得比较清楚,还是不明白的话可以自行百度Java的内存结构。
初始化:执行类的静态变量的初始化代码块和静态代码块
对象的初始化顺序,如上所述,静态变量和静态代码块在对象未初始化时就已经赋初值执行了,而静态变量和静态代码块执行的顺序是按顺序执行的,如果静态变量是在静态代码块后面定义的,则静态代码块先执行。
同样的,对于非静态变量和非静态代码块的执行规则也是按顺序执行的。要注意的是,在JVM加载运行一个类时,它会先加载它的父类,毕竟先有父才有子嘛,嘿嘿,所以我们基本可以得出如下的执行顺序:
父类(静态变量、静态代码块,视顺序执行)--->子类(静态变量、静态代码块)--->new 对象--->父类(成员变量、非静态代码块,视顺序而定)--->父类(构造函数)--->子类(成员变量、非静态代码块)--->子类(构造函数)
所以,看到这,我们大概能猜到一个Java文件编译和执行的过程中发生的一些变化,这个Java文件先被编译成了一个class文件,这里面存储了很多信息,比如版本号、常量等等很多东西,再然后这个class文件被加载到JVM中开始执行了,第一步判断这个class文件数据正不正确,然后为静态变量分配空间,OK,看这道题目,class文件进行加载后,要为静态变量分配空间并赋值,所以public static Test t1 = new Test("t1");先执行,new 对象后,执行成员变量和非静态代码块,所以先执行public int j = print("j");则,
1:j i=0 n=0,此时执行完print后n变为1,i变为1;然后非静态代码块进行执行
2:"构造块" i=1 n=1,此时执行完后i=2,n=2;然后执行构造函数
3:t1 i=2 n=2,此时执行完后i=3,n=3;
同理,public static Test t2 = new Test("t2");开始执行,打印结果如下:
4:j i=3 n=3
5:构造块 i=4 n=4
6:t2 i=5 n=5
接下来:public static int i = print("i");开始执行,要初始化静态变量i,
7:i i=6 n=6
接下来:public static int n = 99;开始执行,它没有输出,只是把n的值变为了99
接下来:static {print("静态块");}开始执行,
8:静态块 i=7 n=99
执行到这,类的初始化工作算是完成了,在main函数里,又new出来一个对象,执行结果就比较简单了,如下:
9:j i=8 n=100
10:构造块 i=9 n=101
11:init i=10 n=102