Java中的類初始化解析
類的初始化
類的初始化是一個Java類生命周期中的其中一個階段。
如下圖所示:
生命周期中的前五個階段(加載、驗證、準(zhǔn)備、解析、初始化)是一個類在JVM中的完整加載過程。
初始化是類加載的最后一個階段,也正是在初始化階段,才會真正開始執(zhí)行類中所寫的Java代碼。
Java虛擬機(jī)規(guī)范中嚴(yán)格規(guī)定了有且只有四種情況必須立即對類進(jìn)行初始化:
1.遇到new、getstatic、putstatic、或者invoicestatic這4條字節(jié)碼指令時,如果類沒有進(jìn)行過初始化,則需要觸發(fā)其初始化。生成這4條指令的最常見的Java代碼場景是:使用new關(guān)鍵字實(shí)例化對象、讀取或設(shè)置一個類的靜態(tài)字段(被final修飾、已在編譯期就把結(jié)果放入常量池的靜態(tài)字段除外)的時候,以及調(diào)用一個類的靜態(tài)方法的時候。
2.使用java.lang.reflect包的方法對類進(jìn)行反射調(diào)用的時候,如果類沒有進(jìn)行過初始化,則需要先觸發(fā)其初始化。
3.當(dāng)初始化一個類的時候,如果發(fā)現(xiàn)父類還沒有進(jìn)行過初始化,需要先觸發(fā)其父類的初始化。
4.當(dāng)虛擬機(jī)啟動時,用戶需要指定一個主類(包含main方法的那個類),虛擬機(jī)會先初始化這個主類。
這四種場景中的行為稱為對一個類進(jìn)行主動引用,除此之外引用類的方式,都不會觸發(fā)初始化,被稱為被動引用。
初始化一個類時,最先執(zhí)行的是靜態(tài)域中代碼,而靜態(tài)域中包含靜態(tài)變量、靜態(tài)塊和靜態(tài)方法,其中需要初始化的是靜態(tài)變量和靜態(tài)塊,這兩者的執(zhí)行順序由代碼的書寫順序決定。
靜態(tài)域中的代碼只會被執(zhí)行一次。如果是使用new關(guān)鍵字實(shí)例化對象,靜態(tài)域中的代碼被執(zhí)行后,緊接著會執(zhí)行構(gòu)造代碼塊和構(gòu)造函數(shù)來實(shí)例化對象,這部分代碼會執(zhí)行多次,每次使用new關(guān)鍵字實(shí)例化對象時,這部分代碼都會執(zhí)行??梢杂萌缦乱欢纬绦蝌炞C這些代碼的執(zhí)行順序:
public class ClassInitTest { public static ClassInitTest c1 = new ClassInitTest(); public static ClassInitTest c2 = new ClassInitTest(); { System.out.println("構(gòu)造代碼塊執(zhí)行"); } static { System.out.println("靜態(tài)代碼塊執(zhí)行"); } public ClassInitTest(){ System.out.println("構(gòu)造方法被執(zhí)行"); } public static void main(String[] args){ System.out.println("main方法的第一行被執(zhí)行"); ClassInitTest c = new ClassInitTest(); System.out.println("main方法的第二行被執(zhí)行"); } }
ClassInitTest是主類,當(dāng)JVM啟動時會對該類進(jìn)行初始化,首先初始化靜態(tài)域中的代碼,可以看到首先是兩個靜態(tài)變量c1和c2,對c1、c2聲明的同時進(jìn)行變量值的初始化,其值是實(shí)例對象,因此構(gòu)造代碼塊和構(gòu)造方法會被執(zhí)行,new了兩次,因此會被執(zhí)行兩次。
緊接著會執(zhí)行靜態(tài)代碼塊中的代碼。靜態(tài)域中的代碼執(zhí)行完畢后便開始執(zhí)行main方法,先是輸出一行語句,接著又實(shí)例化了一個對象,構(gòu)造代碼塊和構(gòu)造方法會被執(zhí)行,最后又輸出一行語句。因此執(zhí)行的結(jié)果如下:
如果存在繼承的情況,初始化子類時,會先初始化父類,同時這幾類代碼執(zhí)行順序的優(yōu)先級保持不變。實(shí)例代碼如下:
public class Father { static { System.out.println("父類的靜態(tài)代碼塊被執(zhí)行"); } { System.out.println("父類的構(gòu)造代碼塊被執(zhí)行"); } public Father(){ System.out.println("父類的構(gòu)造方法被執(zhí)行"); } } public class Child extends Father{ public Child(){ System.out.println("子類構(gòu)造方法被執(zhí)行"); } { System.out.println("子類的構(gòu)造代碼塊被執(zhí)行"); } static { System.out.println("子類的靜態(tài)代碼塊被執(zhí)行"); } public static void main(String [] args){ new Child(); } }
執(zhí)行結(jié)果如下:
之所以子類的構(gòu)造方法執(zhí)行前會先執(zhí)行父類的構(gòu)造方法,是因為子類的構(gòu)造方法總是先調(diào)用父類的某個構(gòu)造方法,也就是說,如果子類沒有明顯地指明使用父類的哪個構(gòu)造方法,子類就調(diào)用父類不帶參數(shù)的構(gòu)造方法。
子類中使用super關(guān)鍵字調(diào)用父類的構(gòu)造方法,而且super必須是子類構(gòu)造方法的第一條語句,如果子類構(gòu)造方法中沒有明顯地寫出super關(guān)鍵字調(diào)用父類的哪個構(gòu)造方法,那么默認(rèn)在子類構(gòu)造方法中有如下語句(即使不寫):
super();//調(diào)用父類的無參構(gòu)造函數(shù)
當(dāng)然也可以在子類的構(gòu)造方法中顯式地用super語句指明調(diào)用父類的哪個構(gòu)造方法(如果父類有多個構(gòu)造方法的話)
super(實(shí)參列表);
到此這篇關(guān)于Java中的類初始化解析的文章就介紹到這了,更多相關(guān)Java類初始化內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot后端數(shù)據(jù)校驗實(shí)戰(zhàn)操作指南
在項?開發(fā)中,對于前端提交的表單,后臺接?接收到表單數(shù)據(jù)后,為了保證程序的嚴(yán)謹(jǐn)性,通常后端會加?業(yè)務(wù)參數(shù)的合法校驗操作來避免程序的?技術(shù)性?bug,這篇文章主要給大家介紹了關(guān)于SpringBoot后端數(shù)據(jù)校驗的相關(guān)資料,需要的朋友可以參考下2022-07-07IntelliJ IDEA本地代碼提交到github網(wǎng)站不顯示與本地不同步問題的解決辦法
今天小編就為大家分享一篇關(guān)于IntelliJ IDEA本地代碼提交到github網(wǎng)站不顯示與本地不同步問題的解決辦法,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2018-10-10Java實(shí)現(xiàn)HttpGet請求傳body參數(shù)
這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)HttpGet請求傳body參數(shù)的相關(guān)知識,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-02-02Java中三種零拷貝的實(shí)現(xiàn)示例以及對比詳解
這篇文章主要介紹了Java中三種零拷貝的實(shí)現(xiàn)示例以及對比詳解,本文主要是介紹幾種零拷貝的實(shí)現(xiàn)示例,以及與最傳統(tǒng)的做一個對比,看看在效率上到底有多大的提升,需要的朋友可以參考下2023-12-12Java 調(diào)用Restful API接口的幾種方式(HTTPS)
這篇文章主要介紹了Java 調(diào)用Restful API接口的幾種方式(HTTPS),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-02-02詳解SpringBoot2.0的@Cacheable(Redis)緩存失效時間解決方案
這篇文章主要介紹了詳解SpringBoot2.0的@Cacheable(Redis)緩存失效時間解決方案,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04SpringMVC源碼解讀之HandlerMapping - AbstractUrlHandlerMapping系列re
這篇文章主要介紹了SpringMVC源碼解讀之HandlerMapping - AbstractUrlHandlerMapping系列request分發(fā) 的相關(guān)資料,需要的朋友可以參考下2016-02-02