Java 初始化执行顺序以及成员变量初始化顺序


一、静态变量初始化顺序

大家先看两个例子:

(1)

 1 public class SingleTon {
 2     public static int count1;
 3     public static int count2 = 1;
 4     private static SingleTon singleTon = new SingleTon();
 5     
 6     private SingleTon() {
 7         count1++;
 8         count2++;
 9     }
10 
11     public static SingleTon getInstance() {
12         return singleTon;
13     }
14 }
15 
16 class Test {
17     public static void main(String[] args) {
18         SingleTon singleTon = SingleTon.getInstance();
19         System.out.println("count1=" + singleTon.count1);
20         System.out.println("count2=" + singleTon.count2);
21     }
22 }

输出

count1=1
count2=2

(2)

 1 public class SingleTon {
 2 
 3     private static SingleTon singleTon = new SingleTon(); //(1)这一步初始化后,count1=1,count2=1
 4     public static int count1; //(2)这一步只是定义了count1变量,并未进行count1初始化,因此count1 =1 count2=1
 5     public static int count2 = 3;//(3)这一步 进行count2初始化,因此将原来的值覆盖,因此count2=3,count1 =1
 6 
 7     private SingleTon() {
 8         count1++; // 此时,count1还未被初始化,因此初始值为0,++后值为1
 9         count2++; // 此时,count2还未被初始化,因此初始值为0,++后值为1
10     }
11 
12     public static SingleTon getInstance() {
13         return singleTon;
14     }
15 }
16 
17 class Test {
18     public static void main(String[] args) {
19         SingleTon singleTon = SingleTon.getInstance();
20         System.out.println("count1=" + singleTon.count1);
21         System.out.println("count2=" + singleTon.count2);
22     }
23 }

count1=1

count2=3

在调用类静态成员(不管是方法还是变量)的时候,按顺序初始化静态属性和代码块,之后才会调用静态方法,非静态成员变量因为没有初始化类,故不会初始化。

二、继承中的初始化

看两个例子:

(1)

 1 class Meal {
 2     Meal() {
 3         System.out.println("Meal()");
 4     }
 5 }
 6 
 7 class Bread {
 8     Bread() {
 9         System.out.println("Bread()");
10     }
11 }
12 
13 class Cheese {
14     Cheese() {
15         System.out.println("Cheese()");
16     }
17 }
18 
19 class Lettuce {
20     Lettuce() {
21         System.out.println("Letuce()");
22     }
23 }
24 
25 class Lunch extends Meal {
26     Lunch() {
27         System.out.println("Lunch()");
28     }
29 }
30 
31 class PortabLunch extends Lunch {
32     PortabLunch() {
33         System.out.println("PortabLunch");
34     }
35 }
36 
37 public class Main extends  PortabLunch{
38     Bread bread = new Bread();
39     Cheese cheese = new Cheese();
40     Lettuce lettuce = new Lettuce();
41 
42     Main() {
43         System.out.println("Main");
44     }
45 
46     public static void main(String[] args) {
47         new Main();
48     }
49 }

输出:

Meal()
Lunch()
PortabLunch
Bread()
Cheese()
Letuce()
Main

说明:子类初始化前要(0)在其他任何事情发生之前,将分配给对象的存储空间初始化为二进制零(1)寻找父类构造器,并且步骤会不断递归,直到找到根类为止。然后自顶向下,逐层调用构造函数,直到最底层的父类构造器。(2)然后按照代码编译顺序依次初始化各成员变量 (3)最后调用本类的构造函数进行初始化。

(2)

 1 public class Glyph {
 2     void draw() {
 3         System.out.println("Glphy draw()");
 4     }
 5 
 6     Glyph() {
 7         System.out.println("Glphy() before draw");
 8         draw();//当子类调用父类构造器的时候,父类其实尚未初始化,因此调用的是子类的draw方法
 9         System.out.println("Glphy() after draw");
10     }
11 }
12 
13 class RoundGlyph extends Glyph {
14     private int radius = 1;
15 
16     RoundGlyph(int r) {
17         radius = r;
18         System.out.println("RoundGlyph.RoundGlyph(),radius:" + radius);
19     }
20 
21     void draw() {
22         System.out.println("RoundGlyph.draw(),radius:" + radius);
23     }
24 }
25 
26 class PolyConstruct {
27     public static void main(String[] args) {
28         new RoundGlyph(5);
29     }
30 }

输出

Glphy() before draw
RoundGlyph.draw(),radius:0
Glphy() after draw
RoundGlyph.RoundGlyph(),radius

说明:在调用父类构造器时,因为draw方法被子类RoundGlyph覆写,因此在Glyph中调用的是子类的draw方法,并且此时由于radius还未进行初始化,因此其值是默认的初始值0.

(3)

 1 public class Glyph {
 2     void draw() {
 3         System.out.println("Glphy draw()");
 4     }
 5 
 6     Glyph() {
 7         System.out.println("Glphy() before draw");
 8         draw();//当子类调用父类构造器的时候,父类其实尚未初始化,因此调用的是子类的draw方法
 9         System.out.println("Glphy() after draw");
10     }
11 }
12 
13 class RoundGlyph extends Glyph {
14     private int radius = 1;
15 
16     RoundGlyph(int r) {
17         radius = r;
18         System.out.println("RoundGlyph.RoundGlyph(),radius:" + radius);
19     }
20 
21 //    void draw() {
22 //        System.out.println("RoundGlyph.draw(),radius:" + radius);
23 //    }
24 }
25 
26 class PolyConstruct {
27     public static void main(String[] args) {
28         new RoundGlyph(5);
29     }
30 }

输出:

Glphy() before draw
Glphy draw()
Glphy() after draw
RoundGlyph.RoundGlyph(),radius:5

 说明:本例子中,子类RoundGlyph中不包含draw方法,因此也就不存在覆写的问题。所以父类初始化时调用的是本类中的draw方法。

通过例子(2)(3)说明,在编写构造器时需要遵循一条准则:"用尽可能简单的方法使得对象进入正常状态;如果可以的话,尽量避免调用其他的方法"。在构造器内能够被唯一安全调用的方法是基类中的final方法或者是private方法,因为它自动属于final方法),这些方法不能被覆盖,因此也不会 出项一些奇怪的问题

下面附一张,JAVA成员变量初始化思维导图,对于理解这一块的知识很有帮助

详细可参考这一文章:https://blog.csdn.net/fly_grass_fish/article/details/81116348   java静态变量static初始化顺序