Java初期化順序の検証(スーパータイプコンストラクターが呼び出される前にXXXを参照できないことによってトリガーされた思考)

1.問題の発見

最近データベースでコーディングしているときに、「スーパータイプコンストラクターが呼び出される前にXXXを参照できません」というエラーメッセージを誤って見つけました。意味は明らかです。この変数は、親クラスのコンストラクターが初期化される前に参照することはできません。startic修飾子がこの変数に追加されると、エラーは報告されなくなります。
ここに写真の説明を挿入

2.推測の検証

では、静的な変更を追加した後、なぜエラーの報告を停止するのですか?明らかに、これはJavaの初期化順序に関連しています。したがって、Coderは、Javaの初期化の順序を検証するための検証コードを作成しました。

/**
 * 父类
 */
public class Base {
    
    

    static String sVar = getString("父类静态变量初始化");
    public String var = getString("父类非静态变量初始化");

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

    {
    
    
        System.out.println("父类的非静态初始化块");
    }

    public Base() {
    
    
        System.out.println("父类构造函数 start");
        draw("父类调用draw方法");//会调用子类覆盖后的方法,这里是null
        System.out.println("父类构造函数 end");
    }

    static String getString(String base) {
    
    
        System.out.println(base);
        return base;
    }

    public void draw(String string) {
    
    
        System.out.println(string);
    }
}
/**
 * 子类
 */
public class SubClass extends Base {
    
    

    public String var = getString("子类初始化非静态变量");
    private String subVar = getString("子类初始化私有变量");
    static String superVar = getString("子类初始化静态变量");

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

    {
    
    
        System.out.println("子类的非静态初始化块");
    }

    SubClass() {
    
    
        System.out.println("子类构造函数start");
        draw("子类调用draw方法");
        System.out.println("子类构造函数end");
    }

    public void draw(String string) {
    
    
        System.out.println(string + subVar);
    }

    public static void main(String[] args) {
    
    
        new SubClass();
    }
}
父类静态变量初始化
父类的静态初始化块
子类初始化静态变量
子类的静态初始化块
父类非静态变量初始化
父类的非静态初始化块
父类构造函数 start
父类调用draw方法 null
父类构造函数 end
子类初始化非静态变量
子类初始化私有变量
子类的非静态初始化块
子类构造函数 start
子类调用draw方法子类初始化私有变量
子类构造函数 end

3.結果分析

この結果から、記事の冒頭でエラーの理由を明確に確認できます。親クラスコンストラクターの初期化は、サブクラスの非静的変数の初期化よりも早く、サブクラスの静的変数の初期化よりも遅くなります。したがって、変数を静的変更に変更しても、エラーは報告されません。Java初期化プロセスを自分で試すことができます。これは、将来のプログラミングで非常に役立ちます。

おすすめ

転載: blog.csdn.net/bigcharsen/article/details/106387682