在网上看了一个这样的题目
public class StaticTest { public static void main(String[] args) { staticFunction(); } static StaticTest st = new StaticTest(); static { System.out.println("1"); } { System.out.println("2"); } StaticTest() { System.out.println("3"); System.out.println("a=" + a + ",b=" + b); } public static void staticFunction() { System.out.println("4"); } int a = 110; static int b = 112; }
问输出顺序是什么?
正确答案是:
2
3
a=110,b=0
1
4
产生这个结果的原因的关键在这一句话:
static StaticTest st = new StaticTest();
st变量的引用是本类的实例,因此在实例化st变量时,将实例初始化嵌入到静态初始化中。因为这一句放在静态初始化的开头,所以static int b=112没有被调用,输出的b=0,同时,输出1也在2和3后面。在对象的初始化时,也是先初始化环境变量,再执行构造函数,a的值为100。
后面我在想把static int b = 112放到st之前会发生什么
public class StaticTest { public static void main(String[] args) { staticFunction(); } static int b = 112; static StaticTest st = new StaticTest(); static { System.out.println("1"); } { System.out.println("2"); } StaticTest() { System.out.println("3"); System.out.println("a=" + a + ",b=" + b); } public static void staticFunction() { System.out.println("4"); } int a = 110; }
执行结果:
2
3
a=110,b=112
1
4
b有值了。所以b在st声明前赋值,st实例化的时候b就有值了。说明静态变量之间实例化是按代码顺序执行的。
如果把静态代码块,提到st之前会发生什么呢?
public class StaticTest { public static void main(String[] args) { staticFunction(); } static { System.out.println("1"); } static StaticTest st = new StaticTest(); { System.out.println("2"); } StaticTest() { System.out.println("3"); System.out.println("a=" + a + ",b=" + b); } public static void staticFunction() { System.out.println("4"); } int a = 110; static int b = 112; }
执行结果:
1
2
3
a=110,b=0
4
发现执行顺序变了,它会先执行静态代码块,再执行st的实体化。
这边,我发现静态代码块与静态成员变量的初始化居然和代码顺序有关
所以不管是静态代码块还是静态成员变量,java都是哪个在前面先执行哪个。