繼承的關鍵字:extends
-
繼承的好處:
1,提高了代碼的復用性。
2,讓類與類之間產生了關系,給第三個特征多態提供了前提.
-
java中支持單繼承。不直接支持多繼承,但對C++中的多繼承機制進行改良。
單繼承:一個子類只能有一個直接父類。
多繼承:一個子類可以有多個直接父類(java中不允許,進行改良)
不直接支持,因為多個父類中有相同成員,會產生調用不確定性。
在java中是通過"多實現"的方式來體現。
- java支持多層(多重)繼承。
C繼承B,B繼承A。
就會出現繼承體系。
- 當要使用一個繼承體系時,
1,查看該體系中的頂層類,了解該體系的基本功能。
2,創建體系中的最子類對象,完成功能的使用。
-
什么時候定義繼承呢?
當類與類之間存在著所屬關系的時候,就定義繼承。xxx是yyy中的一種。 xxx extends yyy
所屬關系: is a 關系。
-
在子父類中,成員的特點體現:
- 成員變量。
- 成員函數。
- 構造函數。
1. 成員變量
當本類的成員和局部變量同名用this區分。
當子父類中的成員變量同名用super區分父類。
this和super的用法很相似,但并不相同。
this:代表一個本類對象的引用。
super:代表一個父類空間。
簡單圖例:
2. 成員函數
當子父類中出現成員函數一模一樣的情況,會運行子類的函數。
這種現象,稱為覆蓋操作。這是函數在子父類中的特性。
函數兩個特性:
1,重載。同一個類中。overload
2,覆蓋。子類中。覆蓋也稱為重寫,覆寫。override
覆蓋注意事項:
1,子類方法覆蓋父類方法時,子類權限必須要大于等于父類的權限。
2,靜態只能覆蓋靜態,或被靜態覆蓋。
什么時候使用覆蓋操作?
當對一個類進行子類的擴展時,子類需要保留父類的功能聲明,
但是要定義子類中該功能的特有內容時,就使用覆蓋操作完成.
3.構造函數
子父類中的構造函數的特點
在子類構造對象時,發現,訪問子類構造函數時,父類也運行了。
為什么呢?
原因是:在子類的構造函數中第一行有一個默認的隱式語句。 super();
子類的實例化過程:子類中所有的構造函數默認都會訪問父類中的空參數的構造函數。
為什么子類實例化的時候要訪問父類中的構造函數呢?
那是因為子類繼承了父類,獲取到了父類中內容(屬性),所以在使用父類內容之前,
要先看父類是如何對自己的內容進行初始化的。
所以子類在構造對象時,必須訪問父類中的構造函數。
為了完成這個必須的動作,就在子類的構造函數中加入了super()語句。
如果父類中沒有定義空參數構造函數,那么子類的構造函數必須用super明確要調用
父類中哪個構造函數。同時子類構造函數中如果使用this調用了本類構造函數時,
那么super就沒有了,因為super和this都只能定義第一行。所以只能有一個。
但是可以保證的是,子類中肯定會有其他的構造函數訪問父類的構造函數。
注意:supre語句必須要定義在子類構造函數的第一行。因為父類的初始化動作要先完成
一個對象實例化過程:
以Person p = new Person();為例
1,JVM會讀取指定的路徑下的Person.class文件,并加載進內存,
并會先加載Person的父類(如果有直接的父類的情況下).
2,在堆內存中的開辟空間,分配地址。
3,并在對象空間中,對對象中的屬性進行默認初始化。
4,調用對應的構造函數進行初始化。
5,在構造函數中,第一行會先到調用父類中構造函數進行初始化。
6,父類初始化完畢后,再對子類的屬性進行顯示初始化。
7,再進行子類構造函數的特定初始化。
8,初始化完畢后,將地址值賦值給引用變量.
示例代碼:
class Fu
{
Fu()
{
super();
show();
return;
}
void show()
{
System.out.println("fu show");
}
}
class Zi extends Fu
{
int num = 8;
Zi()
{
super();
//-->通過super初始化父類內容時,子類的成員變量并未顯示初始化。等super()父類初始化完畢后,
//才進行子類的成員變量顯示初始化。
System.out.println("zi cons run...."+num);
return;
}
void show()
{
System.out.println("zi show..."+num);
}
}
class ExtendsDemo5
{
public static void main(String[] args)
{
Zi z = new Zi();
z.show();
}
}
結果:
zi show...0
zi cons run...8
zi show...8
內存圖示: