類成員變量的初始化


1-1 類成員變量初始化的分類

類成員變量的初始化可簡單分為兩類:非靜態成員變量的初始化(以下簡稱“普通初始化”)和靜態成員變量的初始化(“靜態初始化”)。

1-2 與初始化相關的規律

  1. 訪問類的靜態成員(變量或方法)、首次創建類對象會引發類加載。類加載時會觸發靜態初始化;
  2. 創建類對象才會觸發普通初始化;
  3. 構造函數實際上是靜態方法;
  4. 普通初始化會在任何普通方法(但包括構造函數)被調用前完成(即:普通初始化->構造函數);
  5. 調用靜態方法(除構造函數)只會引發靜態初始化,而不會引發普通初始化;
  6. 調用構造函數會先引發靜態初始化,再引發普通初始化。因為構造函數是靜態方法(調用會引發靜態初始化),又會創建對象(引發普通初始化)。

1-3 類成員變量的初始化順序

  1. 變量(無論是靜態成員變量還是非靜態成員變量)定義的先后順序決定了初始化順序;
  2. 順序優先級:靜態成員變量 = 靜態代碼塊 > 非靜態成員變量 = 非靜態代碼塊 > 構造函數

1-4 相關案例

class Cup{
    Cup(int marker){
        System.out.println("Cup("+marker+")");
    }

    void f1(int marker){
        System.out.println("f1("+marker+")");
    }
}

class Cups{
    static Cup cup1 = new Cup(1);
    Cup cup2 = new Cup(2);
    {
        Cup cup3 = new Cup(3);
    }
    static{
        Cup cup4 = new Cup(4);
    }

    Cups(){
        System.out.println("Cups()");
    }

    static void f2(int marker){
        System.out.println("f2("+marker+")");
    }
}

public class ExplicitStatic {
    public static void main(String[] args) {
        System.out.println("Inside main()");
        Cups.f2(5);
//        Cups cups = new Cups();
    }
}

輸出結果為:

Inside main()
Cup(1)
Cup(4)
f2(5)

Process finished with exit code 0

此結果驗證了1-2中的規律5:調用除構造函數外靜態方法只會引發靜態初始化,而不會引發普通初始化。因為類中的普通成員變量cup2、cup3並未被顯示初始化。

 

public class ExplicitStatic {
    public static void main(String[] args) {
        System.out.println("Inside main()");
//        Cups.f2(5);
        Cups cups = new Cups();
    }
}

輸出的結果為:

Inside main()
Cup(1)
Cup(4)
Cup(2)
Cup(3)
Cups()

Process finished with exit code 0

此結果驗證了1-2中的規律6 以及 初始化順序的優先級。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM