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 一个类的时候,构造器是最后才执行的,这个类的静态变量,静态块以及实例变量要优先于构造器执行。