java什么是繼承意思概念
我們經(jīng)常在項(xiàng)目中使用繼承,但是往往不太明白,程序運(yùn)行的順序以及原理,尤其是使用上轉(zhuǎn)型對(duì)象的時(shí)候,以及父類子類中都有static變量和方法時(shí),不知道先運(yùn)行誰。我也是寫了一個(gè)例子。總結(jié)了一下。
父類:
public class TestStatic {
public static String name="china";
{
System.out.println("========方法體========");
}
static{
name="England";
System.out.println("========靜態(tài)程序塊======");
}
TestStatic(){
System.out.println("=========構(gòu)造方法========");
}
public static void main(String[] args){
System.out.println("========主方法========"+name);
}
public void test(){
System.out.println("========測(cè)試方法=========");
}
}
子類:
public class TestExtendStatic extends TestStatic{
//public static String name="HUBEI";
{
System.out.println("========無名稱方法體========");
}
static{
//name="SUIZHOU";
System.out.println("========子類靜態(tài)程序塊======");
}
TestExtendStatic(){
System.out.println("=========子類構(gòu)造方法========");
}
public void test(){
System.out.println("========子類測(cè)試方法=========");
}
public static void main(String[] args){
System.out.println("========子類主方法========"+name);
TestStatic ts = new TestExtendStatic();// 上轉(zhuǎn)型對(duì)象
ts.test();
}
}
輸出如下:
========靜態(tài)程序塊====== :父類static程序塊
========子類靜態(tài)程序塊====== :子類static程序塊 【不是靜態(tài)方法】
========子類主方法========England :子類主方法
========方法體======== :父類中非靜態(tài)代碼塊
=========構(gòu)造方法======== :父類構(gòu)造方法
========無名稱方法體======== :子類中非靜態(tài)代碼塊
=========子類構(gòu)造方法======== :子類構(gòu)造方法
========子類測(cè)試方法========= :子類測(cè)試方法
執(zhí)行順序: 父類靜態(tài)變量以及靜態(tài)程序塊 --- 子類的靜態(tài)變量以及靜態(tài)程序塊 --- 父類非靜態(tài)代碼塊 --- 父類中構(gòu)造方法 --- 子類中非靜態(tài)代碼塊 --- 子類中構(gòu)造方法 --- 接下來才是 對(duì)象調(diào)用的方
法。
只要是用new 創(chuàng)建對(duì)象,分配了內(nèi)存空間,不管是將引用賦給上轉(zhuǎn)型對(duì)象,還是賦給子類對(duì)象,上面方法都必須執(zhí)行。
即:TestStatic ts = new TestExtendStatic();// 上轉(zhuǎn)型對(duì)象
TestExtendStatic ts = new TestExtendStatic();// 子類對(duì)象
上面加粗程序都會(huì)執(zhí)行。
上面程序中 ts.test(); ts作為上轉(zhuǎn)型對(duì)象調(diào)用的是子類繼承的父類中的方法,因?yàn)閠est()在子類中被重寫了,所以輸出的為子類中的語句。
如果將子類中 main 方法該成如下:
public static void main(String[] args){
System.out.println("========子類主方法========"+name);
TestStatic ts = new TestExtendStatic();
ts.test();
System.out.println("-------------------------");
ts = new TestExtendStatic();
ts.test();
}
輸出:
========靜態(tài)程序塊====== 父類中靜態(tài)程序塊
========子類靜態(tài)程序塊====== 子類中靜態(tài)程序塊
========子類主方法========England 子類中主方法
========方法體======== 父類中非靜態(tài)代碼塊
=========構(gòu)造方法======== 父類中構(gòu)造方法
========無名稱方法體======== 子類中非靜態(tài)程序塊
=========子類構(gòu)造方法======== 子類中構(gòu)造方法
========子類測(cè)試方法========= 對(duì)象具體調(diào)用的方法
------------------------- 靜態(tài)變量以及程序塊只執(zhí)行一次
========方法體======== 父類中非靜態(tài)代碼塊
=========構(gòu)造方法======== 父類中構(gòu)造方法
========無名稱方法體======== 子類中非靜態(tài)代碼塊
=========子類構(gòu)造方法======== 子類中構(gòu)造方法
========子類測(cè)試方法=========
如果將子類主方法 中更改為:
TestStatic ts = new TestStatic ();// 運(yùn)用父類構(gòu)造方法創(chuàng)建
ts.test();
輸出為:
========靜態(tài)程序塊====== 父類靜態(tài)程序塊
========子類靜態(tài)程序塊====== 子類靜態(tài)程序塊【因?yàn)槌绦蛟谧宇愔羞\(yùn)行的,所以子類的靜態(tài)程序塊必須運(yùn)行】
========方法體======== 父類非靜態(tài)程序塊
=========構(gòu)造方法======== 父類構(gòu)造方法
========測(cè)試方法========= 父類具體方法test()
如果將上述代碼放到父類中,就不會(huì)加載子類 靜態(tài)程序塊了。
通過上面 我們還可以發(fā)現(xiàn),靜態(tài)程序塊運(yùn)行 是在主方法之前,非靜態(tài)程序塊運(yùn)行是在主方法之后。
我在父類中 主方法中創(chuàng)建一個(gè)對(duì)象 調(diào)用test(),運(yùn)行的結(jié)果:
========靜態(tài)程序塊====== 靜態(tài)代碼塊
===main==
========方法體======== 非靜態(tài)代碼塊
=========構(gòu)造方法======== 構(gòu)造方法
========測(cè)試方法=========
總結(jié):
程序運(yùn)行時(shí)(一個(gè)類中),會(huì)第一時(shí)間加載運(yùn)行靜態(tài)代碼塊,一旦創(chuàng)建對(duì)象,就會(huì)執(zhí)行非靜態(tài)代碼塊以及無參構(gòu)造方法。 而在繼承中,程序運(yùn)行時(shí)會(huì)先加載父類中靜態(tài)代碼塊 然后加載本身靜態(tài)代碼塊,一旦創(chuàng)建對(duì)象(運(yùn)用子類構(gòu)造方法創(chuàng)建),就會(huì)調(diào)用父類非靜態(tài)代碼塊,父類構(gòu)造方法,然后就是本身 非靜態(tài)代碼塊,本身構(gòu)造方法。