從Java虛擬機角度分析類的實例化順序

1.首先展現一下實例代碼(Son.java & Father.java)java

public class Father {
    
    public static int a=10;//父類的靜態變量
    static{//父類的靜態代碼塊
        a=20;
    }
    {//父類的構造代碼塊
        a=30;
    }
    
    public Father() {//父類的構造方法
        a=40;
    }
}
public class Son extends Father{
    
    public static int s=10;//子類的靜態變量
    public int k=20;//子類的實例變量
    static{//子類的靜態代碼塊
        s=20;
    }
    {//子類的構造代碼塊
        s=30;
    }
    public Son() {//子類的構造函數
        s=40;
    }
    {//子類的構造代碼塊
        s=50;
    }
}

2.將son.java文件編譯爲son.class文件,而後使用javap反編譯查看Son的字節碼指令來分析Son的加載順序,更利於理解(javap -v -c Son > p.txt)。程序員

3.執行代碼"new Son();"後,分析類的加載順序。函數

下面的static{};爲<clinit>函數,son();爲<init>函數。spa

  static {};
    descriptor: ()V
    flags: ACC_STATIC
    Code:
      stack=1, locals=0, args_size=0
         0: bipush        10
         2: putstatic     #11                 // Field s:I--------------------------順序執行靜態變量的賦值
         5: bipush        20
         7: putstatic     #11                 // Field s:I--------------------------順序執行靜態代碼塊
        10: return
  public packet1020.Son();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=2, locals=1, args_size=1
         0: aload_0
         1: invokespecial #16                 // Method packet1020/Father."<init>":()V--------------------執行父類的<init>函數(順序不變,第一個)
         4: aload_0
         5: bipush        20
         7: putfield      #18                 // Field k:I------------------------------------------------按順序收集實例變量賦值
        10: bipush        30
        12: putstatic     #11                 // Field s:I------------------------------------------------按順序收集構造代碼塊
        15: bipush        50
        17: putstatic     #11                 // Field s:I------------------------------------------------按順序收集構造代碼塊
        20: bipush        40
        22: putstatic     #11                 // Field s:I------------------------------------------------最後執行本身的構造函數代碼(順序不變,最後一個)
        25: return

 開始分析:code

1.觸發類的加載,在初始化階段,先執行父類<clinit>函數,而後執行子類<clinit>函數,按照順序執行靜態變量賦值與靜態代碼塊。blog

2.代碼中執行了構造函數,因此執行<init>函數。ip

結論:ci

1.父類中順序執行靜態變量賦值,靜態代碼塊編譯器

2.子類中順序執行靜態變量賦值,靜態代碼塊虛擬機

3.父類中順序執行實例變量賦值,構造代碼塊

4.父類構造函數

5.子類中順序執行實例變量賦值,構造代碼塊

6.子類構造函數

 

名字解釋:摘抄自周志明老師的《深刻理解Java虛擬機:JVM高級特性與最佳實踐》

1.有且只有4中狀況下必須對類進行初始化(執行<clinit>函數)中的第三種:當初始化一個類時,先初始化父類。這就是爲何父類的<clinit>函數先於子類的<clinit>函數執行。

2.<clinit>函數:編譯器按照源代碼中的順序自動收集類中的全部靜態變量的賦值動做和靜態代碼塊中的語句合併而成的。

3.<init>函數:最開始先調用父類的<init>函數,而後編譯器按照源代碼中的順序自動收集類中的實例變量的賦值操做和構造代碼塊中的語句合併,而後插入到構造函數方法前面,最後是程序員本身寫的構造函數代碼。

構造代碼塊執行順序先於構造函數

<init>(){
  1.調用父類<init>方法
  2.順序執行實例變量的賦值操做和構造代碼塊
  3.程序員本身的構造函數方法代碼
}
相關文章
相關標籤/搜索