Java中成员变量、代码块、静态代码块、静态变量和构造函数在加载时的优先级

JAVA初始化顺序:

    1.初始化静态属性,即静态代码和静态变量。按照先父类再子类的顺序初始化。

    2.初始化类属性,即成员变量、代码块和构造函数。其中类变量和代码块优先级相同且均优先于构造函数。也是按照先初始化父类再子类的顺序。

    我们可以看到,类的初始化分为两大部分:即静态属性和非静态属性。静态属性优先于非静态属性。


优先级层次(相同优先级的属性按程序先后顺序初始化):

    第一级:静态属性

        1. 父类:  静态成员变量=静态代码块

        2. 子类:  静态成员变量=静态代码块


    第二级:非静态属性

        1. 父类:

            1.1  成员变量 = 代码块

            1.2  构造函数

        2. 子类:

            2.1  成员变量 = 代码块

            2.2  构造函数


测试代码:

package com.company;
class Father {
    public String str1 = getString("父类public变量");         //类变量
    public static String str2 = getString("父类静态变量");  //静态变量
    private String str3 = getString("父类private变量");        //类变量
    //类代码块
    {
        System.out.println("父类代码块");
    }
    //静态代码块
    static {
        System.out.println("父类静态代码块");
    }
    //father类构造函数
    Father() {
        System.out.println("父类构造函数");
    }
    //输出变量数值以作测试
    public static String getString(String i){
        System.out.println(i);
        return i;
    }
}
class Child extends Father{
    //father类构造函数
    Child() {
        System.out.println("Child类构造函数");
    }
    //类代码块
    {
        System.out.println("子类代码块");
    }

    //静态代码块
    static {
        System.out.println("子类静态代码块");
    }
    private String str6 = getString("子类private变量");   //类变量
    public static String str5 = getString("子类静态变量");  //静态变量
    public String str4 = getString("子类public变量");    //类变量
}
public class Demo {
    public static void main(String[] args) {
        new Child();
    }
}

测试结果:

Java中成员变量、代码块、静态代码块、静态变量和构造函数在加载时的优先级_第1张图片

结果分析:

    在子类的代码中,我们故意将静态代码块、静态变量、成员变量,代码块和构造函数顺序调整。可以看到程序的执行结果始终严格按照上面所列的优先级执行。


/*********延伸**********/

为什么会子类的静态代码出现在父类的非静态属性之前呢,这里涉及到Java的装载和实例化。

类的加载和实例化顺序:

    1) 在堆内存中生成class对象, 把静态变量和静态方法加载到方法区, 这个堆内存中的class对象是方法区数据的入口

    2) 静态变量默认初始化

    3) 静态变量显式初始化

    4) 执行静态代码块

    5) 成员变量默认初始化, 显示初始化

    6) 执行构造函数

    其中有涉及父子关系的类的装载中,先父类静态变量和代码块初始化,再子类。然后实例化时,父类成员变量和代码块初始化, 执行父类构造函数, 子类成员变量和代码块初始化, 执行子类构造函数。




你可能感兴趣的:(JAVA初学)