首先母赵,關(guān)于父類子類靜態(tài)成員的初始化順序,是一個(gè)筆試面試經(jīng)常出現(xiàn)的問(wèn)題具滴,這個(gè)是一個(gè)基礎(chǔ)凹嘲,每一個(gè)寫java代碼的工程師都需要搞清楚的問(wèn)題。但是构韵,搞清楚歸搞清楚周蹭,平時(shí)寫代碼的時(shí)候,正常情況下贞绳,都不會(huì)這樣去寫谷醉,就類似于經(jīng)常考到的 i++ ++i 這樣的問(wèn)題冈闭,平時(shí)俱尼,不會(huì)這樣寫,也不能這樣寫萎攒,代碼審核過(guò)不了的遇八。
這批文章針對(duì)的是剛接觸java不久的新同學(xué)。
首先看下面的代碼
Father類
public class Father {
private int x;
static{
System.out.println("father-static");
}
{
x = 11;
System.out.println("father-member,x = "+x);
}
Father(){
x = 21;
System.out.println("father-constructor,x = "+x);
}
}
Son類耍休,繼承Father
public class Son extends Father {
private int y;
static{
System.out.println("son-static");
}
{
y = 10;
System.out.println("son-member,y="+y);
}
Son(){
y = 20;
System.out.println("son-constructor,y="+y);
}
}
然后寫一個(gè)執(zhí)行類
public class Init {
public static void main(String[] args) {
Son mys = new Son();
System.out.println("over");
}
}
執(zhí)行結(jié)果
father-static
son-static
father-member,x = 11
father-constructor,x = 21
son-member,y=10
son-constructor,y=20
over
通過(guò)結(jié)果來(lái)看刃永,非常清晰。
- 初始化父類的靜態(tài)塊羊精。這一步實(shí)際意思就是初始化父類使用static修飾的成員斯够。
- 初始化子類的靜態(tài)塊。
- 執(zhí)行父類的變量初始化喧锦。
- 執(zhí)行子類的變量初始化读规。
- 執(zhí)行父類的構(gòu)造器。
- 執(zhí)行子類的構(gòu)造器燃少。
這個(gè)過(guò)程束亏,看著代碼理解一下就ok了。
首先阵具,初始化靜態(tài)部分碍遍,這個(gè)是類加載器執(zhí)行的,首先會(huì)把這部分執(zhí)行了阳液。
在初始化初始化子類靜態(tài)變量之前怕敬,會(huì)先初始化父類的靜態(tài)變量,如果子類使用了父類的靜態(tài)變量帘皿,這能保證父類的靜態(tài)變量是初始化好的赖捌。
上面這一步,是在加載類的時(shí)候執(zhí)行的,是在new之前越庇。
然后,執(zhí)行new了奉狈,會(huì)去執(zhí)行類的構(gòu)造函數(shù)卤唉。
子類的構(gòu)造函數(shù)的第一行,是調(diào)用父類的構(gòu)造函數(shù)仁期。
可能有同學(xué)說(shuō)桑驱,我沒(méi)有調(diào)用啊。
對(duì)于無(wú)參構(gòu)造函數(shù)跛蛋,編譯器會(huì)自動(dòng)增加一句 super()
實(shí)際代碼應(yīng)該是這樣
Son(){
super();
y = 20;
System.out.println("son-constructor,y="+y);
}
而另一個(gè)java基本常識(shí)就是熬的,如果我們沒(méi)有寫構(gòu)造函數(shù),編譯器會(huì)自動(dòng)增加一個(gè)無(wú)參構(gòu)造函數(shù)
所以赊级,上面寫的5押框,6步,每次new一個(gè)類的時(shí)候理逊,都是執(zhí)行了的橡伞,只是如果沒(méi)有我們寫的構(gòu)造器,就會(huì)去執(zhí)行一個(gè)空的無(wú)參構(gòu)造器晋被,編譯器自動(dòng)加上去的兑徘。
super() 這一句,會(huì)被自動(dòng)加上去羡洛,我們無(wú)法測(cè)試挂脑,我們可以給Father類的構(gòu)造器,增加一個(gè)參數(shù)欲侮,讓他沒(méi)有無(wú)參構(gòu)造器崭闲。
public class Father {
private int x;
static{
System.out.println("father-static");
}
{
x = 11;
System.out.println("father-member,x = "+x);
}
Father(int i){
x = 21;
System.out.println("father-constructor,x = "+x);
}
}
這個(gè)時(shí)候,去看Son的代碼锈麸,會(huì)發(fā)現(xiàn)镀脂,構(gòu)造器上面有錯(cuò)誤提示。
提示內(nèi)容是
There is no default constructor available in 'com.demo.test1.init.Father'
意思就是忘伞,沒(méi)有默認(rèn)的構(gòu)造器薄翅,也就是沒(méi)有無(wú)參構(gòu)造器。
解決辦法很簡(jiǎn)單氓奈,調(diào)用super的時(shí)候翘魄,把參數(shù)給進(jìn)去。
Son(){
super(1);
y = 20;
System.out.println("son-constructor,y="+y);
}
可以試一下舀奶,在ide里面暑竟,把super(1)放到y(tǒng)=20之后,看看報(bào)錯(cuò)不。
咱們?cè)俎垡晦?br>
首先但荤,類加載器在加載類的時(shí)候罗岖,會(huì)先初始化類的靜態(tài)成員變量或者靜態(tài)塊
這么做的目的,是為了保證腹躁,有些地方直接使用類靜態(tài)成員桑包,這個(gè)時(shí)候,是不會(huì)實(shí)例化這個(gè)類的纺非,也就是不會(huì)觸發(fā)執(zhí)行構(gòu)造器哑了。
然后,實(shí)例化一個(gè)類烧颖,也就是new
一個(gè)類的時(shí)候弱左,會(huì)先初始化普通成員變量及代碼塊,然后再執(zhí)行構(gòu)造器炕淮,這是為了保證構(gòu)造器能夠正常使用這些成員變量拆火,這是一個(gè)類的初始化過(guò)程,如果鳖悠,有父類榜掌,那么先對(duì)父類執(zhí)行這個(gè)過(guò)程,再執(zhí)行子類乘综。
最后憎账,編譯器為我們做了很多事情,例如卡辰,沒(méi)有構(gòu)造器的時(shí)候胞皱,加一個(gè)默認(rèn)無(wú)參的構(gòu)造器;在子類的構(gòu)造器中九妈,默認(rèn)的去調(diào)用父類無(wú)參構(gòu)造器反砌,等等,這些萌朱,都是需要知道的宴树。