举例说明Java中代码块的执行顺序

时间:2022-06-08 17:31:25

前言
    今天在看Android ContentProvider实现的时候,突然想到了Java类在new的过程中,静态域、静态块、非静态域、非静态块、构造函数的执行顺序问题。其实这是一个很经典的问题,非常考察对Java基础知识的掌握程度。很多面试过程中相信也有这样的问题,趁着周末有时间复习一下。

结论
    这里先把整理好的结论抛给大家,然后我在写个程序来验证我们的结论。在Java类被new的过程中,执行顺序如下:

  •     实现自身的静态属性和静态代码块。(根据代码出现的顺序决定谁先执行)
  •     实现自身的非静态属性和非静态代码块。
  •     执行自身的构造函数。

    在实现继承的类被new的过程中,初始化执行顺序如下:

  •     实现父类的公共静态属性和静态块级代码。
  •     实现自身的静态属性和静态块级代码。
  •     实现父类的非静态属性和非静态代码块。
  •     执行父类的构造函数。
  •     实现自身的非静态属性和非静态代码块。
  •     执行自身的构造函数。

    这里需要简单的介绍一下静态代码块和非静态代码块。
1. 静态代码块:

    static { 
    } 

2. 非静态代码块

    { 
    } 

    静态代码块和非静态代码块的异同点如下:

  •     相同点:都是JVM加载类时且在构造函数执行之前执行,在类中都可以定义多个,一般在代码块中对一些static变量进行赋值。
  •     不同点:静态代码块在非静态代码块之前执行(静态代码块 > 非静态代码块)。静态代码块只在第一次new时执行一次,之后不再执行。而非静态代码块每new一次就执行一次。


验证
    对于结论的最好验证就是写出代码来进行结果证明。首先,来看一下无继承的类初始化时的执行顺序,代码如下:

 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
public class InitOderTest {
  public static String STATIC_FIELD = "静态属性";
   
  // 静态块
  static {
    System.out.println(STATIC_FIELD);
    System.out.println("静态代码块");
  }
   
  public String field = "非静态属性";
   
  // 非静态块
  {
    System.out.println(field);
    System.out.println("非静态代码块");
  }
 
  public InitOderTest() {
    System.out.println("无参构造函数");
  }
   
  public static void main(String[] args) {
    InitOderTest test = new InitOderTest();
  }
}

    执行结果:

  •     静态属性 
  •     静态代码块 
  •     非静态属性 
  •     非静态代码块 
  •     无参构造函数 

    接下来,我们验证一下,当Java类实现继承后,执行顺序是否和我们的结论吻合。测试代码如下:

   

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
class ParentTest {
   public static String PARENT_STATIC_FIELD = "父类-静态属性";
  
   // 父类-静态块
   static {
     System.out.println(PARENT_STATIC_FIELD);
     System.out.println("父类-静态代码块");
   }
  
   public static String parentField = "父类-非静态属性";
  
   // 父类-非静态块
   {
     System.out.println(parentField);
     System.out.println("父类-非静态代码块");
   }
  
   public ParentTest() {
     System.out.println("父类—无参构造函数");
   }
 }
  
 public class InitOderTest extends ParentTest {
   public static String STATIC_FIELD = "静态属性";
  
   // 静态块
   static {
     System.out.println(STATIC_FIELD);
     System.out.println("静态代码块");
   }
  
   public String field = "非静态属性";
  
   // 非静态块
   {
     System.out.println(field);
     System.out.println("非静态代码块");
   }
  
   public InitOderTest() {
     System.out.println("无参构造函数");
   }
  
   public static void main(String[] args) {
     InitOderTest test = new InitOderTest();
   }
 }

    执行结果如下:

  •     父类-静态属性 
  •     父类-静态代码块 
  •     静态属性 
  •     静态代码块 
  •     父类-非静态属性 
  •     父类-非静态代码块 
  •     父类—无参构造函数 
  •     非静态属性 
  •     非静态代码块 
  •     无参构造函数