java的static塊執行時機

berber78發表於2015-06-12
一、誤區:簡單認為JAVA靜態程式碼塊在類被載入時就會自動執行。證錯如下:  

class MyClass1 {
    static {//靜態塊
        System.out.println("static block ");
    }
}
public class Main {

    Class[] classArray = {
            MyClass1.class//這樣引用該類,必然需要將該類載入到虛擬機器中
    };
    public static void main(String[] args){
        System.out.println("hello word");
    }

}
執行結果:並沒有輸出" static bolck"

二、正解:static塊真正的執行時機。如果瞭解JVM原理,我們知道,一個類的執行分為以下步驟:

  1. 裝載
  2. 連線
  3. 初始化

    其中裝載階段又三個基本動作組成

  1.     通過型別的完全限定名,產生一個代表該型別的二進位制資料流
  2.     解析這個二進位制資料流為方法區內的內部資料結
  3.     構建立一個表示該型別的java.lang.Class類的例項

    另外如果一個類裝載器在預先裝載的時遇到缺失或錯誤的class檔案,它需要等到程式首次主動使用該類時才報告錯誤。

 

    連線階段又分為三部分

  1. 驗證,確認型別符合Java語言的語義,檢查各個類之間的二進位制相容性(比如final的類不用擁有子類等),另外還需要進行符號引用的驗證。
  2. 準備,Java虛擬機器為類變數分配記憶體,設定預設初始值。
  3. 解析(可選的) ,在型別的常量池中尋找類,介面,欄位和方法的符號引用,把這些符號引用替換成直接引用的過程。

   

  當一個類被主動使用時,Java虛擬就會對其初始化,如下六種情況為主動使用

  1. 當建立某個類的新例項時(如通過new或者反射,克隆,反序列化等)
  2. 當呼叫某個類的靜態方法時
  3. 當使用某個類或介面的靜態欄位時
  4. 當呼叫Java API中的某些反射方法時,比如類Class中的方法,或者java.lang.reflect中的類的方法時
  5. 當初始化某個子類時
  6. 當虛擬機器啟動某個被標明為啟動類的類(即包含main方法的那個類)

    Java編譯器會收集所有的類變數初始化語句和型別的靜態初始化器,將這些放到一個特殊的方法中:clinit。 

 
實際上,static塊的執行發生在“初始化”的階段。初始化階段,jvm主要完成對靜態變數的初始化,靜態塊執行等工作。

下面我們看看執行static塊的幾種情況:

1、第一次new A()的過程會列印"";因為這個過程包括了初始化

2、第一次Class.forName("A")的過程會列印"";因為這個過程相當於Class.forName("A",true,this.getClass().getClassLoader());

3、第一次Class.forName("A",false,this.getClass().getClassLoader())的過程則不會列印""。因為false指明瞭裝載類的過程中,不進行初始化。不初始化則不會執行static塊。


 參考資料:《深入Java虛擬機器》


相關文章