舉例說明Java中代碼塊的執(zhí)行順序
前言
今天在看Android ContentProvider實現(xiàn)的時候,突然想到了Java類在new的過程中,靜態(tài)域、靜態(tài)塊、非靜態(tài)域、非靜態(tài)塊、構(gòu)造函數(shù)的執(zhí)行順序問題。其實這是一個很經(jīng)典的問題,非??疾鞂ava基礎(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變量進行賦值。
- 不同點:靜態(tài)代碼塊在非靜態(tài)代碼塊之前執(zhí)行(靜態(tài)代碼塊 > 非靜態(tài)代碼塊)。靜態(tài)代碼塊只在第一次new時執(zhí)行一次,之后不再執(zhí)行。而非靜態(tài)代碼塊每new一次就執(zhí)行一次。
驗證
對于結(jié)論的最好驗證就是寫出代碼來進行結(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項目如何配置多數(shù)據(jù)源(已上生產(chǎn),親測有效)
這篇文章主要介紹了spring項目如何配置多數(shù)據(jù)源(已上生產(chǎn),親測有效),具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-12-12
SpringBoot通過@Scheduled實現(xiàn)定時任務(wù)及單線程運行問題解決
Scheduled定時任務(wù)是Spring boot自身提供的功能,所以不需要引入Maven依賴包,下面這篇文章主要給大家介紹了關(guān)于SpringBoot通過@Scheduled實現(xiàn)定時任務(wù)以及問題解決的相關(guān)資料,需要的朋友可以參考下2023-02-02
Java?CompletableFuture實現(xiàn)多線程異步編排
這篇文章主要為大家介紹了Java?CompletableFuture實現(xiàn)多線程異步編排,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-09-09

