一、 代码块的概念
在探究对象初始化顺序之前,我们先通过代码来了解一下代码块的概念。
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