父类子类的静态初始化块,初始化块,构造器执行顺序

时间:2022-04-10 19:32:00

class Parent {
public static String p_StaticField = "父类--静态变量";
public String p_Field = "父类--变量";
// 代码块
{
System.out.println(p_Field);
System.out.println("父类--初始化块");
}

// 构造函数
public Parent() {
System.out.println("父类--构造器");
}

// 静态初始化块
static {
System.out.println(p_StaticField);
System.out.println("父类--静态初始化块");
}
}

public class StaticBlock extends Parent {
public static String s_StaticField = "子类--静态变量";
public String s_Field = "子类--变量";
{
System.out.println(s_Field);
System.out.println("子类--初始化块");
}

// 静态初始化块
static {
System.out.println(s_StaticField);
System.out.println("子类--静态初始化块");
}

public StaticBlock() {
System.out.println("子类--构造器");
}

public static void main(String[] args) {
System.out.println("-------------------");
new Parent();
System.out.println("-------------------");
new StaticBlock();
System.out.println("-------------------");
new StaticBlock();
}
}

执行结果:

父类子类的静态初始化块,初始化块,构造器执行顺序

执行顺序规则:
对于静态变量、静态初始化块、变量、初始化块、构造器,它们的初始化顺序依次是:
(静态变量、静态初始化块)>(变量、初始化块)> 构造器。
对于静态变量和静态初始化块之间、变量和初始化块之间的先后顺序又是怎样呢?
是否静态变量总是先于静态初始化块,变量总是先于初始化块就被初始化了呢?
实际上这取决于它们在类中出现的先后顺序。
注意,变量必须先定义再使用,所以静态变量、变量必须放在使用它们的静态初始化块、初始化块前面!


执行过程如下:
程序执行始点是SubClass.main,
1、执行SubClass的main之前,
要看SubClass中是否有静态的变量和语句,如果有,先给这些静态的变量分配存储空间和执行静态语句(不是静态方法),且由于SubClass的父类中也有静态的变量,根据继承的特性,则先执行父类Parent的静态数据的初始化,然会执行子类的静态数据的初始化。
父类--静态变量  
父类--静态初始化块  
子类--静态变量  
子类--静态初始化块  
2、执行main方法中的new Parent(); 语句,进行Parent的类的实例化
因为Parent的静态数据已经实例化,并且在一个执行过程只实例化一次,所以在执行new Parenet()语句时,
先执行非静态变量定义和类的非静态语句,之后再执行构造方法,所以有下面的结果
父类--变量  
父类--初始化块  
父类--构造器  
3、在执行main方法中的new SubClass(); 语句,进行SubClass的类的实例化
同2,父类和子类的静态数据不执行,因为SubClass继承Parent,所以首先执行父类的非静态的变量和类语句的执行,在调用SubClass的构造方法之前,如果没有明确的说明,则先执行父类的同参数的构造方法,然后执行子类的实例化,则出现下面的结果
子类--变量  
子类--初始化块  

子类--构造器  

----------------------------------------------------------------------------------------------------

1.静态初始化块只在某个类第一次装入内存时才执行,因此只有第一次new Parent()才有“静态初始化块”输出,而后面却不执行输出。
2.初始化块在某个对象生成时(被new)执行。所以两次new SubClass()都执行初始化块。因为SubClass也是Parent类,所有new SubClass()时也执行Parent的初始化块
----------------------------------------------------------------------------------------------------
总结如下:
(1)先静态。具体是先父静态->子静态(参考下面2)

(2)先父后子。先父的全部,然后子的全部。
(3)优先级:父类 > 子类 。静态代码块 > 非静态代码块 > 构造函数 (与位置的前后无关系)

先静态,可以这么理解,只要出现了某类的关键字,那么这个类的类变量或方法就要初始化,否则引用的时候就没有初始化的时机了;
先父后子,可以这样理解,子类会引用父类的东西,那么必须先父后子。
----------------------------------------------------------------------------------------------------
在运行SubClass(因为main方法)时JVM首先会装载SubClass,(其中的步骤省略。。)
之后会初始化SubClass类的初始值(程序员在程序中赋予的值)
所有静态变量的初始化都会被JAVA编译器放在一个特殊的方法中称之为类初始化方法,只能被JVM调用。
此时会判断是否存在直接超类,如果还未被初始化,直接初始化超类的静态变量包括块。(主动初始化)
所以SubClass的超类Parent的静态变量p_StaticField和块首先被初始化,(注意如果声明是final的就不会初始化它),如果Parent还有超类的话在此前就应该先初始化Parent的静态变量,显然Object类没有静态变量和块。
然后初始化SubClass的静态变量s_StaticField和静态块
静态变量初始化完成后开始初始化成员变量(new Parent() new为主动初始化)和块
因为静态变量和块在类装载的时候初始化的,与具体的实例无关,所以下面的new SubClass(); 只能初始化自身类的成员变量和块,却不能初始化超类的静态变量和块,当然初始化顺序永远是超类先被初始化。

----------------------------------------------------------------------------------------------------
http://topic.csdn.net/u/20090722/10/b18e34dd-abb6-48cc-9370-456e9c427003.html
java面试解惑系列