Java 中变量的加载顺序
开始学习这一块知识点之前,需要知道优快云博客: JAVA中静态块、静态变量加载顺序,这里直接做总结:
如果类还没有被加载:
1、先执行父类的静态代码块和静态变量初始化,并且静态代码块和静态变量的执行顺序只跟代码中出现的顺序有关。
2、执行子类的静态代码块和静态变量初始化。
3、执行父类的实例变量初始化
4、执行父类的构造函数
5、执行子类的实例变量初始化
6、执行子类的构造函数
如果类已经被加载:
则静态代码块和静态变量就不用重复执行,再创建类对象时,只执行与实例相关的变量初始化和构造方法。
然后我们直接看几个例子:
例一
public class Test1 {
int a = 1;
int b = getNumberOfB();
int getNumberOfB() {
return this.a + 10;
}
public static void main(String[] args) {
Test1 test1 = new Test1();
System.out.println("test1.b = " + test1.b);
}
}
结果
test1.b = 11
例二
public class Test2 {
int b = getNumberOfB();
int a = 1;
int getNumberOfB() {
return this.a + 10;
}
public static void main(String[] args) {
Test2 test2 = new Test2();
System.out.println("test2.b = " + test2.b);
}
}
结果
test2.b = 10
解释
由于JVM
执行new Test2();
这一步时,会先初始化变量b,然后初始化变量a,最后执行Test2
这个类的默认构造器。在初始化b的时候去执行方法getNumberOfB()
,这个方法里面变量this.a
的值是按照JVM
默认提供的初值0来的,而不是按照上面代码中的int a = 1;
,这是因为JVM
的执行流程当前处于为b赋值的阶段,即语句int b = getNumberOfB();
中,还没有执行到下一条语句int a = 1;
,故方法getNumberOfB()
中的变量this.a
只能按照JVM
提供的默认初值来处理。
例三
public class Test3 {
int b = getNumberOfB();
int getNumberOfB() {
return a + 10;
}
int a = 1;
public static void main(String[] args) {
Test3 test3 = new Test3();
System.out.println("test3.b = " + test3.b);
}
}
结果
test3.b = 10
解释
原因与例二相同。
例四
public class Test4 {
int a = 1;
public Test4(int a) {
this.a = a;
}
int b = getNumberOfB();
int getNumberOfB() {
return this.a + 10;
}
public static void main(String[] args) {
Test4 test4 = new Test4(50);
System.out.println("test4.a = " + test4.a);
System.out.println("test4.b = " + test4.b);
}
}
结果
test4.a = 50
test4.b = 11
解释
记住new 一个类的时候,构造器是最后才执行的,这个类的静态变量,静态块以及实例变量要优先于构造器执行。