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初期化プロセスを自分で試すことができます。これは、将来のプログラミングで非常に役立ちます。