我們經常在項目中使用繼承,但是往往不太明白,程序運行的順序以及原理,尤其是使用上轉型對象的時候,以及父類子類中都有static變量和方法時,不知道先運行誰。我也是寫了一個例子。總結了一下。
代碼如下:
父類:
public class TestStatic {
public static String name="china";
{
System.out.println("========方法體========");
}
static{
name="England";
System.out.println("========靜態程序塊======");
}
TestStatic(){
System.out.println("=========構造方法========");
}
public static void main(String[] args){
System.out.println("========主方法========"+name);
}
public void test(){
System.out.println("========測試方法=========");
}
}
子類:
public class TestExtendStatic extends TestStatic{
//public static String name="HUBEI";
{
System.out.println("========無名稱方法體========");
}
static{
//name="SUIZHOU";
System.out.println("========子類靜態程序塊======");
}
TestExtendStatic(){
System.out.println("=========子類構造方法========");
}
public void test(){
System.out.println("========子類測試方法=========");
}
public static void main(String[] args){
System.out.println("========子類主方法========"+name);
TestStatic ts = new TestExtendStatic();// 上轉型對象
ts.test();
}
}
輸出如下:
========靜態程序塊====== :父類static程序塊
========子類靜態程序塊====== :子類static程序塊 【不是靜態方法】
========子類主方法========England :子類主方法
========方法體======== :父類中非靜態代碼塊
=========構造方法======== :父類構造方法
========無名稱方法體======== :子類中非靜態代碼塊
=========子類構造方法======== :子類構造方法
========子類測試方法========= :子類測試方法
執行順序: 父類靜態變量以及靜態程序塊 --- 子類的靜態變量以及靜態程序塊 --- 父類非靜態代碼塊 --- 父類中構造方法 --- 子類中非靜態代碼塊 --- 子類中構造方法 --- 接下來才是 對象調用的方
法。
只要是用new 創建對象,分配了內存空間,不管是將引用賦給上轉型對象,還是賦給子類對象,上面方法都必須執行。
即:TestStatic ts = new TestExtendStatic();// 上轉型對象
TestExtendStatic ts = new TestExtendStatic();// 子類對象
上面加粗程序都會執行。
上面程序中 ts.test(); ts作為上轉型對象調用的是 子類繼承的父類中的方法,因為test()在子類中被重寫了,所以輸出的為子類中的語句。
如果將子類中 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();
}
輸出:
========靜態程序塊====== 父類中靜態程序塊
========子類靜態程序塊====== 子類中靜態程序塊
========子類主方法========England 子類中主方法
========方法體======== 父類中非靜態代碼塊
=========構造方法======== 父類中構造方法
========無名稱方法體======== 子類中非靜態程序塊
=========子類構造方法======== 子類中構造方法
========子類測試方法========= 對象具體調用的方法
------------------------- 靜態變量以及程序塊只執行一次
========方法體======== 父類中非靜態代碼塊
=========構造方法======== 父類中構造方法
========無名稱方法體======== 子類中非靜態代碼塊
=========子類構造方法======== 子類中構造方法
========子類測試方法=========
如果將子類主方法 中更改為:
代碼如下:
TestStatic ts = new TestStatic ();// 運用父類構造方法創建
ts.test();
輸出為:
========靜態程序塊====== 父類靜態程序塊
========子類靜態程序塊====== 子類靜態程序塊 【因為程序在子類中運行的,所以子類的靜態程序塊必須運行】
========方法體======== 父類非靜態程序塊
=========構造方法======== 父類構造方法
========測試方法========= 父類具體方法test()
如果將上述代碼放到父類中,就不會加載子類 靜態程序塊了。
通過上面 我們還可以發現,靜態程序塊運行 是在主方法之前,非靜態程序塊運行是在主方法之後。
我在父類中 主方法中創建一個對象 調用test(),運行的結果:
========靜態程序塊====== 靜態代碼塊
===main==
========方法體======== 非靜態代碼塊
=========構造方法======== 構造方法
========測試方法=========
總結:
程序運行時(一個類中),會第一時間加載運行靜態代碼塊,一旦創建對象,就會執行非靜態代碼塊以及無參構造方法。 而在繼承中,程序運行時 會先加載父類中靜態代碼塊 然後加載本身靜態代碼塊,一旦創建對象(運用子類構造方法創建),就會調用 父類非靜態代碼塊,父類構造方法,然後就是本身 非靜態代碼塊,本身構造方法。