靜態(tài)代碼塊,構(gòu)造代碼塊,構(gòu)造函數(shù)執(zhí)行執(zhí)行順序詳解

Java提高篇——靜態(tài)代碼塊、構(gòu)造代碼塊、構(gòu)造函數(shù)以及Java類初始化順序

閱讀目錄

構(gòu)造函數(shù)

構(gòu)造代碼塊

靜態(tài)代碼塊

Java類初始化順序

靜態(tài)代碼塊:用staitc聲明,jvm加載類時(shí)執(zhí)行,僅執(zhí)行一次

構(gòu)造代碼塊:類中直接用{}定義,每一次創(chuàng)建對(duì)象時(shí)執(zhí)行。

執(zhí)行順序優(yōu)先級(jí):靜態(tài)塊,main(),構(gòu)造塊,構(gòu)造方法。

構(gòu)造函數(shù)

public HelloA(){//構(gòu)造函數(shù)}

關(guān)于構(gòu)造函數(shù),以下幾點(diǎn)要注意:

1.對(duì)象一建立,就會(huì)調(diào)用與之相應(yīng)的構(gòu)造函數(shù),也就是說,不建立對(duì)象,構(gòu)造函數(shù)時(shí)不會(huì)運(yùn)行的。

2.構(gòu)造函數(shù)的作用是用于給對(duì)象進(jìn)行初始化。

3.一個(gè)對(duì)象建立,構(gòu)造函數(shù)只運(yùn)行一次,而一般方法可以被該對(duì)象調(diào)用多次。

構(gòu)造代碼塊

{//構(gòu)造代碼塊? ? }

關(guān)于構(gòu)造代碼塊,以下幾點(diǎn)要注意:

構(gòu)造代碼塊的作用是給對(duì)象進(jìn)行初始化。

對(duì)象一建立就運(yùn)行構(gòu)造代碼塊了,而且優(yōu)先于構(gòu)造函數(shù)執(zhí)行。這里要強(qiáng)調(diào)一下,有對(duì)象建立,才會(huì)運(yùn)行構(gòu)造代碼塊,類不能調(diào)用構(gòu)造代碼塊的,而且構(gòu)造代碼塊與構(gòu)造函數(shù)的執(zhí)行順序是前者先于后者執(zhí)行。

構(gòu)造代碼塊與構(gòu)造函數(shù)的區(qū)別是:構(gòu)造代碼塊是給所有對(duì)象進(jìn)行統(tǒng)一初始化,而構(gòu)造函數(shù)是給對(duì)應(yīng)的對(duì)象初始化,因?yàn)闃?gòu)造函數(shù)是可以多個(gè)的,運(yùn)行哪個(gè)構(gòu)造函數(shù)就會(huì)建立什么樣的對(duì)象,但無論建立哪個(gè)對(duì)象,都會(huì)先執(zhí)行相同的構(gòu)造代碼塊。也就是說,構(gòu)造代碼塊中定義的是不同對(duì)象共性的初始化內(nèi)容。

靜態(tài)代碼塊

static{//靜態(tài)代碼塊? ? }

關(guān)于靜態(tài)代碼塊,要注意的是:

它是隨著類的加載而執(zhí)行,只執(zhí)行一次,并優(yōu)先于主函數(shù)。具體說,靜態(tài)代碼塊是由類調(diào)用的。類調(diào)用時(shí),先執(zhí)行靜態(tài)代碼塊,然后才執(zhí)行主函數(shù)的。

靜態(tài)代碼塊其實(shí)就是給類初始化的,而構(gòu)造代碼塊是給對(duì)象初始化的。

靜態(tài)代碼塊中的變量是局部變量,與普通函數(shù)中的局部變量性質(zhì)沒有區(qū)別。

一個(gè)類中可以有多個(gè)靜態(tài)代碼塊

public class Test{

staitc int cnt=6;

static{

? ? ? cnt+=9;

}

public static void main(String[] args) {

? ? ? System.out.println(cnt);

}

static{

? ? ? cnt/=3;

}

}

運(yùn)行結(jié)果:5

java類初始化順序

## 對(duì)于一個(gè)類的情況

例子1:

public class HelloA {

? ? public HelloA(){

??? //構(gòu)造函數(shù)System.out.println("A的構(gòu)造函數(shù)");? ?

? ? }

? ? {

???? //構(gòu)造代碼塊System.out.println("A的構(gòu)造代碼塊");? ?

? ? }

? ? static{

???? //靜態(tài)代碼塊System.out.println("A的靜態(tài)代碼塊");? ? ? ?

? ? }

? ? public static void main(String[] args) {

? ? }

}

運(yùn)行結(jié)果:

A的靜態(tài)代碼塊


例子2:

public class HelloA {

? ? public HelloA(){

??? //構(gòu)造函數(shù)System.out.println("A的構(gòu)造函數(shù)");? ?

? ? }

? ? {

??? //構(gòu)造代碼塊System.out.println("A的構(gòu)造代碼塊");? ?

? ? }

? ? static{

?? //靜態(tài)代碼塊System.out.println("A的靜態(tài)代碼塊");? ? ? ?

? ? }

? ? public static void main(String[] args) {

? ? ? ? HelloA a=new HelloA();? ?

? ? }

}

運(yùn)行結(jié)果:

A的靜態(tài)代碼塊

A的構(gòu)造代碼塊

A的構(gòu)造函數(shù)

例子3:

public class HelloA {

? ? public HelloA(){??

?? //構(gòu)造函數(shù)System.out.println("A的構(gòu)造函數(shù)");? ?

? ? }

? ? {

??? //構(gòu)造代碼塊System.out.println("A的構(gòu)造代碼塊");? ?

? ? }

? ? static{

??? //靜態(tài)代碼塊System.out.println("A的靜態(tài)代碼塊");? ? ? ?

? ? }

? ? public static void main(String[] args) {

? ? ? ? HelloA a = new HelloA();

? ? ? ? HelloA b = new HelloA();

? ? }

}

運(yùn)行結(jié)果:

A的靜態(tài)代碼塊

A的構(gòu)造代碼塊

A的構(gòu)造函數(shù)

A的構(gòu)造代碼塊

A的構(gòu)造函數(shù)

對(duì)于一個(gè)類而言,按照如下順序執(zhí)行:

執(zhí)行靜態(tài)代碼塊

執(zhí)行構(gòu)造代碼塊

執(zhí)行構(gòu)造函數(shù)

對(duì)于靜態(tài)變量、靜態(tài)初始化塊、變量、初始化塊、構(gòu)造器,它們的初始化順序依次是(靜態(tài)變量、靜態(tài)初始化塊)>(變量、初始化塊)>構(gòu)造器。

例子4:


運(yùn)行以上代碼,我們會(huì)得到如下的輸出結(jié)果:

靜態(tài)變量

靜態(tài)初始化塊

變量

初始化塊

構(gòu)造器


運(yùn)行結(jié)果:

A的靜態(tài)代碼塊

B的靜態(tài)代碼塊

A的構(gòu)造代碼塊

A的構(gòu)造函數(shù)

B的構(gòu)造代碼塊

B的構(gòu)造函數(shù)

當(dāng)涉及到繼承時(shí),按照如下順序執(zhí)行:

1. 執(zhí)行父類的靜態(tài)代碼塊,并初始化父類靜態(tài)成員變量

2. 執(zhí)行子類的靜態(tài)代碼塊,并初始化子類靜態(tài)成員變量

3. 執(zhí)行父類的構(gòu)造代碼塊,執(zhí)行父類的構(gòu)造函數(shù),并初始化父類普通成員變量

4. 執(zhí)行子類的構(gòu)造代碼塊, 執(zhí)行子類的構(gòu)造函數(shù),并初始化子類普通成員變量



結(jié)果:

父類--靜態(tài)變量

父類--靜態(tài)初始化塊

子類--靜態(tài)變量

子類--靜態(tài)初始化塊

子類main方法

父類--變量

父類--初始化塊

父類--構(gòu)造器

i=9, j=0

子類--變量

子類--初始化塊

子類--構(gòu)造器

i=9,j=20

子類的靜態(tài)變量和靜態(tài)初始化塊的初始化是在父類的變量、初始化塊和構(gòu)造器初始化之前就完成了。靜態(tài)變量、靜態(tài)初始化塊,變量、初始化塊初始化了順序取決于它們?cè)陬愔谐霈F(xiàn)的先后順序。

### 分析

(1)訪問SubClass.main(),(這是一個(gè)static方法),于是裝載器就會(huì)為你尋找已經(jīng)編譯的SubClass類的代碼(也就是SubClass.class文件)。在裝載的過程中,裝載器注意到它有一個(gè)基類(也就是extends所要表示的意思),于是它再裝載基類。不管你創(chuàng)不創(chuàng)建基類對(duì)象,這個(gè)過程總會(huì)發(fā)生。如果基類還有基類,那么第二個(gè)基類也會(huì)被裝載,依此類推。

(2)執(zhí)行根基類的static初始化,然后是下一個(gè)派生類的static初始化,依此類推。這個(gè)順序非常重要,因?yàn)榕缮惖摹皊tatic初始化”有可能要依賴基類成員的正確初始化。

(3)當(dāng)所有必要的類都已經(jīng)裝載結(jié)束,開始執(zhí)行main()方法體,并用new SubClass()創(chuàng)建對(duì)象。

(4)類SubClass存在父類,則調(diào)用父類的構(gòu)造函數(shù),你可以使用super來指定調(diào)用哪個(gè)構(gòu)造函數(shù)?;惖臉?gòu)造過程以及構(gòu)造順序,同派生類的相同。首先基類中各個(gè)變量按照字面順序進(jìn)行初始化,然后執(zhí)行基類的構(gòu)造函數(shù)的其余部分。

(5)對(duì)子類成員數(shù)據(jù)按照它們聲明的順序初始化,執(zhí)行子類構(gòu)造函數(shù)的其余部分。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容