C++繼承與派生的概念、什么是繼承和派生
在C++中可重用性是通過繼承(inheritance)這一機制來實現的。因此,繼承是C++的一個重要組成部分。
前面介紹了類,一個類中包含了若干數據成員和成員函數。在不同的類中,數據成員和成員函數是不相同的。但有時兩個類的內容基本相同或有一部分相同,例如巳聲明了學生基本數據的類Student:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
class Student { public : void display( ) //對成員函數display的定義 { cout<< "num: " <<num<<endl; cout<< "name: " << name <<endl; cout << "sex: " <<sex<<endl; } private : int num; string name; char sex; }; |
如果學校的某一部門除了需要用到學號、姓名、性別以外,還需要用到年齡、地址等信息。當然可以重新聲明另一個類class Student1:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
class Student1 { public : void display( ) //此行原來已有 { cout<< "num: " <<num<<endl; //此行原來已有 cout<< "name: " << name <<endl; //此行原來已有 cout << "sex: " <<sex<<endl; //此行原來已有 cout << "age: " <<age<<endl; cout << "address: " <<addr<<endl; } private : int num; //此行原來已有 string name; //此行原來已有 char sex; //此行原來已有 int age; char addr[20]; }; |
可以看到有相當一部分是原來已經有的,可以利用原來聲明的類Student作為基礎,再加上新的內容即可,以減少重復的工作量。C++提供的繼承機制就是為了解決這個問題。
在C++中,所謂“繼承”就是在一個已存在的類的基礎上建立一個新的類。已存在的類稱為“基類(base class)”或“父類(father class)”,新建的類稱為“派生類(derived class)”或“子類(son class )”。
一個新類從已有的類那里獲得其已有特性,這種現象稱為類的繼承。通過繼承,一個新建子類從已有的父類那里獲得父類的特性。從另一角度說,從已有的類(父類)產生一個新的子類,稱為類的派生。類的繼承是用已有的類來建立專用類的編程技術。派生類繼承了基類的所有數據成員和成員函數,并可以對成員作必要的增加或調整。一個基類可以派生出多個派生類,每一個派生類又可以作為基類再派生出新的派生類,因此基類和派生類是相對而言的。一代一代地派生下去,就形成類的繼承層次結構。相當于一個大的家族,有許多分支,所有的子孫后代都繼承了祖輩的基本特征,同時又有區別和發展。與之相仿,類的每一次派生,都繼承了其基類的基本特征,同時又根據需要調整和擴充原 有的特征。
以上介紹的是最簡單的情況:一個派生類只從一個基類派生,這稱為單繼承(single inheritance),這種繼承關系所形成的層次是一個樹形結構,如圖所示。
一個派生類不僅可以從一個基類派生,也可以從多個基類派生。也就是說,一個派生類可以有一個或者多個基類。一個派生類有兩個或多個基類的稱為多重繼承(multiple inheritance)。關于基類和派生類的關系,可以表述為派生類是基類的具體化,而基類則是派生類的抽象。
C++派生類成員的訪問屬性
既然派生類中包含基類成員和派生類自己增加的成員,就產生了這兩部分成員的關系和訪問屬性的問題。在建立派生類的時候,并不是簡單地把基類的私有成員直接作為派生類的私有成員,把基類的公用成員直接作為派生類的公用成員。
實際上,對基類成員和派生類自己增加的成員是按不同的原則處理的。具體說,在討論訪問屬性時,要考慮以下幾種情況:
基類的成員函數訪問基類成員。
派生類的成員函數訪問派生類自己增加的成員。
基類的成員函數訪問派生類的成員。
派生類的成員函數訪問基類的成員。
在派生類外訪問派生類的成員。
在派生類外訪問基類的成員。
對于第(1)和第(2)種情況,比較簡單,基類的成員函數可以訪問基類成員,派生類的成員函數可以訪問派生類成員。私有數據成員只能被同一類中的成員函數訪問,公用成員可以被外界訪問。
第(3)種情況也比較明確,基類的成員函數只能訪問基類的成員,而不能訪問派生類的成員。
第(5)種情況也比較明確,在派生類外可以訪問派生類的公用成員,而不能訪問派生類的私有成員。
對于第(4)和第(6)種情況,就稍微復雜一些,也容易混淆。譬如,有人提出這樣的問題:
基類中的成員函數是可以訪問基類中的任一成員的,那么派生類中新增加的成員是否可以同樣地訪問基類中的私有成員;
在派生類外,能否通過派生類的對象名訪問從基類繼承的公用成員。
這些牽涉到如何確定基類的成員在派生類中的訪問屬性的問題,不僅要考慮對基類成員所聲明的訪問屬性,還要考慮派生類所聲明的對基類的繼承方式,根據這兩個因素共同決定基類成員在派生類中的訪問屬性。
前面已提到,在派生類中,對基類的繼承方式可以有public(公用的)、private (私有的)和protected(保護的)3種。不同的繼承方式決定了基類成員在派生類中的訪問屬性。簡單地說可以總結為以下幾點。
1) 公用繼承(public inheritance)
基類的公用成員和保護成員在派生類中保持原有訪問屬性,其私有成員仍為基類私有。
2) 私有繼承(private inheritance)
基類的公用成員和保護成員在派生類中成了私有成員,其私有成員仍為基類私有。
3) 受保護的繼承(protected inheritance)
基類的公用成員和保護成員在派生類中成了保護成員,其私有成員仍為基類私有。保護成員的意思是,不能被外界引用,但可以被派生類的成員引用。