1. 程式人生 > >從Java虛擬機角度分析類的實例化順序

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

構造代碼塊 on() 觸發 進行 cal utf The father 實例代碼

1.首先展示一下實例代碼(Son.java & Father.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>函數。

  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

開始分析:

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

2.代碼中執行了構造函數,所以執行<init>函數。

結論:

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

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

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

4.父類構造函數

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

6.子類構造函數

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

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

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

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

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

<init>(){
  1.調用父類<init>方法
  2.順序執行實例變量的賦值操作和構造代碼塊
  3.程序員自己的構造函數方法代碼
}

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