舉例說明Java中代碼塊的執(zhí)行順序
前言
今天在看Android ContentProvider實現(xiàn)的時候,突然想到了Java類在new的過程中,靜態(tài)域、靜態(tài)塊、非靜態(tài)域、非靜態(tài)塊、構(gòu)造函數(shù)的執(zhí)行順序問題。其實這是一個很經(jīng)典的問題,非常考察對Java基礎(chǔ)知識的掌握程度。很多面試過程中相信也有這樣的問題,趁著周末有時間復(fù)習(xí)一下。
結(jié)論
這里先把整理好的結(jié)論拋給大家,然后我在寫個程序來驗證我們的結(jié)論。在Java類被new的過程中,執(zhí)行順序如下:
- 實現(xiàn)自身的靜態(tài)屬性和靜態(tài)代碼塊。(根據(jù)代碼出現(xiàn)的順序決定誰先執(zhí)行)
- 實現(xiàn)自身的非靜態(tài)屬性和非靜態(tài)代碼塊。
- 執(zhí)行自身的構(gòu)造函數(shù)。
在實現(xiàn)繼承的類被new的過程中,初始化執(zhí)行順序如下:
- 實現(xiàn)父類的公共靜態(tài)屬性和靜態(tài)塊級代碼。
- 實現(xiàn)自身的靜態(tài)屬性和靜態(tài)塊級代碼。
- 實現(xiàn)父類的非靜態(tài)屬性和非靜態(tài)代碼塊。
- 執(zhí)行父類的構(gòu)造函數(shù)。
- 實現(xiàn)自身的非靜態(tài)屬性和非靜態(tài)代碼塊。
- 執(zhí)行自身的構(gòu)造函數(shù)。
這里需要簡單的介紹一下靜態(tài)代碼塊和非靜態(tài)代碼塊。
1. 靜態(tài)代碼塊:
static {
}
2. 非靜態(tài)代碼塊
{
}
靜態(tài)代碼塊和非靜態(tài)代碼塊的異同點如下:
- 相同點:都是JVM加載類時且在構(gòu)造函數(shù)執(zhí)行之前執(zhí)行,在類中都可以定義多個,一般在代碼塊中對一些static變量進(jìn)行賦值。
- 不同點:靜態(tài)代碼塊在非靜態(tài)代碼塊之前執(zhí)行(靜態(tài)代碼塊 > 非靜態(tài)代碼塊)。靜態(tài)代碼塊只在第一次new時執(zhí)行一次,之后不再執(zhí)行。而非靜態(tài)代碼塊每new一次就執(zhí)行一次。
驗證
對于結(jié)論的最好驗證就是寫出代碼來進(jìn)行結(jié)果證明。首先,來看一下無繼承的類初始化時的執(zhí)行順序,代碼如下:
public class InitOderTest { public static String STATIC_FIELD = "靜態(tài)屬性"; // 靜態(tài)塊 static { System.out.println(STATIC_FIELD); System.out.println("靜態(tài)代碼塊"); } public String field = "非靜態(tài)屬性"; // 非靜態(tài)塊 { System.out.println(field); System.out.println("非靜態(tài)代碼塊"); } public InitOderTest() { System.out.println("無參構(gòu)造函數(shù)"); } public static void main(String[] args) { InitOderTest test = new InitOderTest(); } }
執(zhí)行結(jié)果:
- 靜態(tài)屬性
- 靜態(tài)代碼塊
- 非靜態(tài)屬性
- 非靜態(tài)代碼塊
- 無參構(gòu)造函數(shù)
接下來,我們驗證一下,當(dāng)Java類實現(xiàn)繼承后,執(zhí)行順序是否和我們的結(jié)論吻合。測試代碼如下:
class ParentTest { public static String PARENT_STATIC_FIELD = "父類-靜態(tài)屬性"; // 父類-靜態(tài)塊 static { System.out.println(PARENT_STATIC_FIELD); System.out.println("父類-靜態(tài)代碼塊"); } public static String parentField = "父類-非靜態(tài)屬性"; // 父類-非靜態(tài)塊 { System.out.println(parentField); System.out.println("父類-非靜態(tài)代碼塊"); } public ParentTest() { System.out.println("父類—無參構(gòu)造函數(shù)"); } } public class InitOderTest extends ParentTest { public static String STATIC_FIELD = "靜態(tài)屬性"; // 靜態(tài)塊 static { System.out.println(STATIC_FIELD); System.out.println("靜態(tài)代碼塊"); } public String field = "非靜態(tài)屬性"; // 非靜態(tài)塊 { System.out.println(field); System.out.println("非靜態(tài)代碼塊"); } public InitOderTest() { System.out.println("無參構(gòu)造函數(shù)"); } public static void main(String[] args) { InitOderTest test = new InitOderTest(); } }
執(zhí)行結(jié)果如下:
- 父類-靜態(tài)屬性
- 父類-靜態(tài)代碼塊
- 靜態(tài)屬性
- 靜態(tài)代碼塊
- 父類-非靜態(tài)屬性
- 父類-非靜態(tài)代碼塊
- 父類—無參構(gòu)造函數(shù)
- 非靜態(tài)屬性
- 非靜態(tài)代碼塊
- 無參構(gòu)造函數(shù)
相關(guān)文章
spring boot請求異常處理并返回對應(yīng)的html頁面
這篇文章主要介紹了spring boot處理請求異常并返回對應(yīng)的html頁面,包括404異常處理和500異常處理,需要的朋友可以參考下2017-07-07IntelliJ IDEA 2019.2 x64的安裝、應(yīng)用與簡單配置(圖文)
這篇文章主要介紹了IntelliJ IDEA 2019.2 x64的安裝、應(yīng)用與簡單配置,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-10-10spring+html5實現(xiàn)安全傳輸隨機數(shù)字密碼鍵盤
這篇文章主要為大家詳細(xì)介紹了spring html5實現(xiàn)安全傳輸隨機數(shù)字密碼鍵盤,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-04-04MyEclipse設(shè)置Console輸出到文件的實現(xiàn)方法
下面小編就為大家?guī)硪黄狹yEclipse設(shè)置Console輸出到文件的實現(xiàn)方法。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-07-07