Java——对象初始化顺序使用详解

时间:2022-09-23 14:41:07

一、 代码块的概念

在探究对象初始化顺序之前,我们先通过代码来了解一下代码块的概念。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
class Test{
  public static String str1; //静态字段
 
  public String str2;     //普通字段
 
  static{          
    //静态代码块
  }
 
  {
    //构造代码块
  }
 
  public Test() {
    //构造函数
  }
}

二、 创建子类对象时,对象的初始化顺序

1. 字段初始化、代码块和构造函数的执行顺序

我们先看代码和结果

?
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
public class CodeBlockTest {
  public static void main(String[] args) {
    Child child = new Child();
  }
}
class Father {
  public static String fatherStr1 = "fatherStr1(静态字段初始化值)";
 
  public String fatherStr2 = "fatherStr2(字段初始化值)";
  static {
    System.out.println("父类静态代码块:" + fatherStr1);
    fatherStr1 = "fatherStr1(静态代码块赋值)";
  }
  {
    System.out.println("父类构造代码块:" + fatherStr2);
    fatherStr2 = "fatherStr2(构造代码块赋值)";
  }
  public Father() {
    System.out.println("父类构造函数块:" + fatherStr2);
    fatherStr2 = "fatherStr2(构造函数赋值)";
  }
}
class Child extends Father {
  public static String childStr1 = "childStr1(静态字段初始化值)";
  public String childStr2 = "childStr2(字段初始化值)";
  static {
    System.out.println("子类静态代码块:" + childStr1);
    childStr1 = "childStr1(静态代码块赋值)";
  }
  {
    System.out.println("子类构造代码块:" + childStr2);
    childStr2 = "childStr2(构造代码块赋值)";
  }
  public Child() {
    System.out.println("子类构造函数:" + childStr2);
    childStr2 = "childStr2(构造函数赋值)";
  }
}
//  输出结果:
//  父类静态代码块:fatherStr1(静态字段初始化值)
//  子类静态代码块:childStr1(静态字段初始化值)
//  父类构造代码块:fatherStr2(字段初始化值)
//  父类构造函数块:fatherStr2(构造代码块赋值)
//  子类构造代码块:childStr2(字段初始化值)
//  子类构造函数:childStr2(构造代码块赋值)

通过每执行一个代码块或构造函数,输出字段在上一代码块执行后的值,以此来探究对象的初始化顺序。

由目前的输出结果可知,对于对象的初始化顺序,我们可以得出以下结论:

1. 父类静态字段初始化

2. 父类静态代码块、子类静态字段初始化 (接下来探究两者的顺序)

3. 子类静态代码块

4. 父类普通字段初始化

5. 父类构造代码块

6. 父类构造函数

7. 子类普通字段初始化

8. 子类构造代码块

9. 子类构造函数

2. 父类静态代码块和子类静态字段初始化的执行顺序

还是一样,我们通过代码的执行结果来探究两者间的执行顺序。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
public class CodeBloacTest2 {
  public static void main(String[] args) {
    Child child = new Child();
  }
}
class Father {
  public static String fatherStr = "(静态字段初始化值)";
  static {
    System.out.println("父类静态代码块:fatherStr" + fatherStr);
    fatherStr = "(静态代码块赋值)";
  }
}
class Child extends Father {
  public static String childStr = fatherStr;
  static {
    System.out.println("子类静态代码块:childStr = fatherStr" + childStr);
    childStr = "(静态代码块赋值)";
  }
}
//  输出结果:
//  父类静态代码块:fatherStr(静态字段初始化值)
//  子类静态代码块:childStr = fatherStr(静态代码块赋值)

我们在子类静态字段childStr初始化的时候,赋的是父类静态字段fatherStr的值。由输出结果可知,childStr初始化后的值是父类静态代码块执行后赋予fatherStr的值。由此可知两者的执行顺序为:父类静态代码块==>子类静态字段初始化

三、 结论

父类静态字段初始化

父类静态代码块

子类静态字段初始化

子类静态代码块

父类普通字段初始化

父类构造代码块

父类构造函数

子类普通字段初始化

子类构造代码块

子类构造函数

通过结论我们可以很明显的看出:static字段、代码块的执行顺序优先于非static字段、代码块。这是因为在静态域是属于类的,在类加载后就一直存在;而普通域需要创建对象才能访问。而在创建对象时,需要先加载父类,然后再加载子类,因此父类的静态字段初始化和静态代码块执行先于子类。

以上内容希望对各位朋友有所帮助

原文链接:http://blog.csdn.net/lim_dev/article/details/70161726