版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報或認(rèn)領(lǐng)
文檔簡介
1、第四章繼承和派生-4.1繼承和派生的概念面向?qū)ο蟪绦蛟O(shè)計有4個主要特點: 抽象、封裝、繼承和多態(tài)性。在本章中主要介紹有關(guān)繼承的知識,在下一章中將介紹多態(tài)性。C+語言提供了類的繼承機制,解決了軟件重用問題。4.1.1 繼承與派生的概念一個類中包含了若干數(shù)據(jù)成員和成員函數(shù)。在不同的類中,數(shù)據(jù)成員和成員函數(shù)是不相同的。但有時兩個類的內(nèi)容基本相同或有一部分相同。-類的繼承:一個新類從已存在的類那里獲得該類已有的特性叫作類的繼承。已存在的類叫作父類,也叫作基類。產(chǎn)生的新類叫作子類或派生類。類的派生:從一個已有的類那里產(chǎn)生一個新類的過程叫類的派生。已存在的類叫作父類,也叫作基類。產(chǎn)生的新類叫作派生類或子類
2、。類的繼承和派生是同一概念,前者是從子類的角度來說,后者是從父類的角度來說的。我們通常說子類繼承了父類。父類派生了子類。-描述各級學(xué)生的類的繼承關(guān)系如下圖:基類與派生類的關(guān)系:派生類是基類的具體化,基類則是派生類的抽象一個派生類的對象也是一個基類的對象。應(yīng)該具有基類的一切屬性和方法。派生類除了具有基類的一切屬性和方法外,還可以有自己所特有的屬性和方法。4.1.2 派生類和基類的關(guān)系-4.1.3 單繼承與多繼承單繼承:一個派生類只從一個基類繼承。多重繼承:一個派生類從兩個或多個基類繼承。-4.2 派生類的聲明方式聲明派生類的一般形式為class 派生類名: 繼承方式 基類名派生類新增加的成員 ;
3、繼承方式包括: public(公有的),private(私有的)和protected(受保護的),此項是可選的,如果不寫此項,則默認(rèn)為private(私有的)。如下程序演示了類Rectangle(四邊形)由類Point繼承而來。-void SetPoint(int x,int y)this-x=x;this-y=y;void MovePoint(int dx,int dy)x+=dx;y+=dy;void ShowPoint()cout(x,y);/Point.h#include #include using namespace std;class Pointprivate:int x,y;p
4、ublic:int GetX()return x;int GetY()return y;-void SetRect(int x,int y,int w,int h)SetPoint(x,y);width=w;height=h;void ShowRect()cout左上角坐標(biāo)為:;ShowPoint();coutendl;cout寬為:widthendl;cout長為:heightendl;/Rectangle.h#include #include Point.husing namespace std;class Rectangle:public Pointprivate:int width;i
5、nt height;public:int GetWidth()return width;int GetHight()return height;-/main.cpp#include Rectangle.hvoid main()Rectangle r;r.SetRect(0,0,10,20);r.ShowRect();r.MovePoint(10,10);r.ShowRect();-4.3 派生類的構(gòu)成派生類中的成員包括從基類繼承過來的成員和自己增加的成員兩大部分。在基類中包括數(shù)據(jù)成員和成員函數(shù)(或稱數(shù)據(jù)與方法)兩部分,派生類分為兩大部分: 一部分是從基類繼承來的成員,另一部分是在聲明派生類時增
6、加的部分。每一部分均分別包括數(shù)據(jù)成員和成員函數(shù)。-如果在派生類中定義了和基類中同名函數(shù)(函數(shù)參數(shù)個數(shù)和類型可以相同也可以不相同),則派生類中的函數(shù)會隱藏基類的同名函數(shù)。在派生類中不能直接訪問基類中的同名函數(shù)。(注意與重載的區(qū)別。在同一個類中的同名不同參函數(shù)為重載函數(shù))如程序PointRect1所示:-4.4繼承方式派生類的繼承方式有三種:public,private,protected。不同的繼承方式?jīng)Q定了基類成員在派生類中的訪問屬性。繼承方式基類中的訪問屬性派生類中的訪問屬性publicpublicpublicprotectedprotectedprivate不可訪問privatepubli
7、cprivateprotectedprivateprivate不可訪問protectedpublicprotectedprotectedprotectedprivate不可訪問-4.4.1類的保護成員前面介紹過類的成員(數(shù)據(jù)成員和成員函數(shù))的訪問屬性有私有的(private)的和公有的(public的)。另外還提到類的訪問屬性也可以有保護的(protected的)。類中的protected成員與private成員一樣,只能在本類的成員函數(shù)中訪問,不能在類外通過對象來訪問。但通過上面的表中可以看出當(dāng)類派生時,基類的private成員在派生類中是不可訪問的。而基類的protected成員在派生類中
8、隨繼承方式的不同而不同。-class Drived:public Baseprotected:int j;public:void Fun()i=20;#include using namespace std;class Baseprotected:int i;public:void F();void main()Drived d;-4.5 派生類的構(gòu)造函數(shù)和析構(gòu)函數(shù)構(gòu)造函數(shù)的主要作用是對數(shù)據(jù)成員初始化。在設(shè)計派生類的構(gòu)造函數(shù)時,不僅要考慮派生類所增加的數(shù)據(jù)成員的初始化,還應(yīng)當(dāng)考慮基類的數(shù)據(jù)成員初始化。也就是說,希望在執(zhí)行派生類的構(gòu)造函數(shù)時,使派生類的數(shù)據(jù)成員和基類的數(shù)據(jù)成員同時都被初始化。解決
9、這個問題的思路是: 在執(zhí)行派生類的構(gòu)造函數(shù)時,調(diào)用基類的構(gòu)造函數(shù)注意:派生類繼承基類的除構(gòu)造函數(shù)和析構(gòu)函數(shù)以外的所有函數(shù)。-4.4.1 簡單的派生類的構(gòu)造函數(shù)簡單的派生類:只有一個基類,而且只有一級派生(只有直接派生類,沒有間接派生類),在派生類的數(shù)據(jù)成員中不包含基類的對象(即子對象)。簡單派生類中我們一般采用在派生類的構(gòu)造函數(shù)初始化列表中調(diào)用基類的構(gòu)造函數(shù)來對繼承基類的數(shù)據(jù)成員進(jìn)行初始化。其一般形式為:派生類構(gòu)造函數(shù)名(總參數(shù)表列): 基類構(gòu)造函數(shù)名(參數(shù)表列) 派生類中新增數(shù)據(jù)成員初始化語句、簡單派生類的構(gòu)造函數(shù)的形式-#include #includeusing namespace st
10、d;class Student public: Student(int n,string nam,char s) num=n;name=nam;sex=s; Student( ) protected: int num; string name; char sex ; ;-class Student1: public Student public: Student1(int n,string nam,char s,int a,string ad):Student(n,nam,s) age=a; addr=ad;void show( )coutnum: numendl;coutname: name
11、endl;coutsex: sexendl;coutage: ageendl;coutaddress: addrendlendl;Student1( ) private: int age; string addr; ;-在main函數(shù)中,建立對象stud1時指定了5個實參。它們按順序傳遞給派生類構(gòu)造函數(shù)Student1的形參。然后,派生類構(gòu)造函數(shù)將前面3個傳遞給基類構(gòu)造函數(shù)的形參。-、簡單派生類的構(gòu)造函數(shù)的幾點說明a.定義派生類的對象時系統(tǒng)自動調(diào)用派生類構(gòu)造函數(shù)之前會先調(diào)用其基類的構(gòu)造函數(shù)?;惖臉?gòu)造函數(shù)是在派生類的構(gòu)造函數(shù)的初始化列表中給出。如果在初始化列表中沒顯式給出調(diào)用語句則調(diào)用基類的默
12、認(rèn)構(gòu)造函數(shù)。 例如前例中派生類的構(gòu)造函數(shù)為:Student1(int n,string nam,char s,int a,string ad):Student(n,nam,s)如果在main函數(shù)中定義一個Student1類的對象時。系統(tǒng)會先調(diào)用基類的構(gòu)造函數(shù)。然后執(zhí)行Student1的構(gòu)造函數(shù)體內(nèi)的代碼完成對派生類成員的構(gòu)造。如果Student1類的構(gòu)造函數(shù)改為Student1(int n,string nam,char)則會先調(diào)用基類的默認(rèn)構(gòu)造函數(shù)。-.當(dāng)派生類構(gòu)造函數(shù)在類外定義時,則只在類外的函數(shù)定義處加上調(diào)用基類的初始化列表。在類內(nèi)申明的地方不加.由前面例題中的構(gòu)造函數(shù)Student1(
13、int n,string nam,char s,int a,string ad):Student(n,nam,s)可以看出派生類的構(gòu)造函數(shù)的初始化列表中是在調(diào)用基類的構(gòu)造函數(shù)而不是在申明或定義基類的構(gòu)造函數(shù)。所以Student1中的五個參數(shù)是形參(帶參數(shù)類型),而其初始化列表中的Student的三個參數(shù)是實參(不帶有類型),這些實參取自Student1,所以Studen中的三個參數(shù)也可以為常數(shù)。例如可將Student1的構(gòu)造函數(shù)改為: Student1(string nam,char s,int a,string ad):Student(10010,nam,s);.在派生類對象釋放時先執(zhí)行派生
14、類的析構(gòu)函數(shù),然后執(zhí)行基類的析構(gòu)函數(shù)。-例題:定義一個點類Point.由Point派生出一個圓類Circle/Point.h文件#ifndef POINT_H#define POINT_Hclass Pointprotected:float x;float y;public:Point()x=0; y=0;Point(float x,float y);void Show();#endif/Point.cpp文件#include #include Point.husing namespace std;Point:Point(float x,float y)this-x=x;this-y=y;vo
15、id Point:Show()cout(x,y)endl;-/Circle.h文件#ifndef CIRCLE_H#define CIRCLE_H#include #include Point.husing namespace std;class Circle:public Pointprotected:float r;public:Circle(float x,float y,float r);void Show();float GetArea();float GetLength();#endif/Circle.cpp文件#include #include CirCle.husing nam
16、espace std;Circle:Circle(float x,float y,float r):Point(x,y)this-r=r;void Circle:Show()cout圓心為:;Point(x,y).Show();cout半徑為:rendl;float Circle:GetArea()return 3.14159*r*r;float Circle:GetLength()return 3.14159*2*r;-4.5.2 有子對象的派生類的構(gòu)造函數(shù)類的數(shù)據(jù)成員中還可以包含類對象。例如前面的Student1類繼承自Student類,我們可以在Student1類中加入一個Student
17、類的對象來表示該同學(xué)所在班的班長。如下程序所示:#include #include using namespace std;class Studentpublic: Student(int n, string nam ) num=n;name=nam; void display( ) coutnum:numendlname:nameendl;protected: int num; string name;-class Student1: public Student public:Student1(int n, string nam,int n1, string nam1,int a, str
18、ing ad):Student(n,nam),monitor(n1,nam1) age=a; addr=ad;void show( )cout學(xué)生信息為:endl;display(); coutage: ageendl; coutaddress: addrendl; cout班長為:endl;monitor.display( );private: Student monitor; int age; string addr;-int main( )Student1 stud1(10010,Wang-li,10001,Li-sun,19,115 Beijing Road,Shanghai);stu
19、d1.show( ); return 0;派生類構(gòu)造函數(shù)的任務(wù)應(yīng)該包括3個部分: (1) 對基類數(shù)據(jù)成員初始化;(2) 對子對象數(shù)據(jù)成員初始化;(3) 對派生類數(shù)據(jù)成員初始化。其中前兩個必須放在派生類的構(gòu)造函數(shù)的初始化列表中進(jìn)行,第(3)個可以在函數(shù)體中也可以在初始化列表中進(jìn)行。-定義派生類構(gòu)造函數(shù)的一般形式為派生類構(gòu)造函數(shù)名(總參數(shù)表列): 基類構(gòu)造函數(shù)名(參數(shù)表列),子對象名(參數(shù)表列) 派生類中新增數(shù)成員據(jù)成員初始化語句執(zhí)行派生類構(gòu)造函數(shù)的順序是: 調(diào)用基類構(gòu)造函數(shù),對基類數(shù)據(jù)成員初始化; 調(diào)用子對象構(gòu)造函數(shù),對子對象數(shù)據(jù)成員初始化; 再執(zhí)行派生類構(gòu)造函數(shù)本身,對派生類數(shù)據(jù)成員初始化。以
20、上次序是固定的,不會因為基類構(gòu)造函數(shù)調(diào)用寫在前面還是子對象名寫在前面而改變。-例題:定義一個點類Point.由Point派生出一個圓類Circle/Point.h文件#ifndef POINT_H#define POINT_Hclass Pointprotected:float x;float y;public:Point()x=0; y=0;Point(float x,float y);void Show();#endif/Point.cpp文件#include #include Point.husing namespace std;Point:Point(float x,float y)t
21、his-x=x;this-y=y;void Point:Show()cout(x,y)endl;4.5.3 多層派生時的構(gòu)造函數(shù)-/Circle.h文件#ifndef CIRCLE_H#define CIRCLE_H#include #include Point.husing namespace std;class Circle:public Pointprotected:float r;public:Circle(float x,float y,float r);void Show();float GetArea();float GetLength();#endif/Circle.cpp文件
22、#include #include CirCle.husing namespace std;Circle:Circle(float x,float y,float r):Point(x,y)this-r=r;void Circle:Show()cout圓心為:;Point(x,y).Show();cout半徑為:rh=h;void Show();float GetArea();float GetVolume();#endif-/Column.cpp#include Column.h#include using namespace std;void Column:Show()Circle:Sho
23、w();cout高為:hendl;float Column:GetArea()return 2*Circle:GetArea()+GetLength()*h;float Column:GetVolume()return Circle:GetArea()*h;-在多層派生的情況下:派生類的構(gòu)造函數(shù)初始化列表中只須寫出其上一層派生類的構(gòu)造函數(shù),不要再寫上其間接子類的構(gòu)造函數(shù)。-4.5.4 派生類的析構(gòu)函數(shù)在派生時,派生類是不能繼承基類的析構(gòu)函數(shù)的,也需要通過派生類的析構(gòu)函數(shù)去調(diào)用基類的析構(gòu)函數(shù)。在派生類中可以根據(jù)需要定義自己的析構(gòu)函數(shù),用來對派生類中所增加的成員進(jìn)行清理工作。基類的清理工作仍然由基
24、類的析構(gòu)函數(shù)負(fù)責(zé)。在執(zhí)行派生類的析構(gòu)函數(shù)時,系統(tǒng)會自動調(diào)用基類的析構(gòu)函數(shù)和子對象的析構(gòu)函數(shù),對基類和子對象進(jìn)行清理。調(diào)用的順序與構(gòu)造函數(shù)正好相反: 先執(zhí)行派生類自己的析構(gòu)函數(shù),對派生類新增加的成員進(jìn)行清理,然后調(diào)用子對象的析構(gòu)函數(shù),對子對象進(jìn)行清理,最后調(diào)用基類的析構(gòu)函數(shù),對基類進(jìn)行清理。-4.6 多重繼承前面討論的是單繼承,即一個類是從一個基類派生而來的。實際上,常常有這樣的情況: 一個派生類有兩個或多個基類,派生類從兩個或多個基類中繼承所需的屬性。C+為了適應(yīng)這種情況,允許一個派生類同時繼承多個基類。這種行為稱為多重繼承(multiple inheritance)。4.6.1 聲明多重繼承
25、的方式聲多重繼承子類的方法和單繼承相似,只是在標(biāo)明子類的位置將繼承的父類都寫上,且以豆號隔開。例如類多重繼承了類A,B,C則申明類的方法如下:class D:public A,protected B,private C類新增加的成員-多重繼承的子類具有多個父類,子類中具有所有父類的所有成員。且對多個父類可以有不同的繼承方式,不同的繼承方式?jīng)Q定了繼承而來的父類的成員在子類中的訪問屬性的不同。4.6.2 多重繼承的派生類的構(gòu)造函數(shù)多重繼承派生類的構(gòu)造函數(shù)形式與單繼承時的構(gòu)造函數(shù)形式基本相同,只是在初始列表中包含多個基類構(gòu)造函數(shù)。形式如下:派生類構(gòu)造函數(shù)名(總參數(shù)表列): 基類1構(gòu)造函數(shù)(參數(shù)表列)
26、, 基類2構(gòu)造函數(shù)(參數(shù)表列), 基類3構(gòu)造函數(shù) (參數(shù)表列) 派生類中新增數(shù)據(jù)成員成員初始化語句-派生類構(gòu)造函數(shù)的執(zhí)行順序同樣為: 先調(diào)用基類的構(gòu)造函數(shù),再執(zhí)行派生類構(gòu)造函數(shù)的函數(shù)體。調(diào)用基類構(gòu)造函數(shù)的順序是按照聲明派生類時基類出現(xiàn)的順序,與構(gòu)造函數(shù)初始化列表中基類的排列順序無關(guān)。#include #include using namespace std;class Teacherpublic: Teacher(string nam,int a, string t) name=nam;age=a;title=t;void display( ) coutname:nameendl;coutag
27、eageendl;couttitle:titleendl;-protected: string name; int age; string title; ;class Student public:Student(string nam,char s,float sco)name1=nam;sex=s;score=sco; void display1( ) coutname:name1endl;coutsex:sexendl;coutscore:scoreendl;-protected: string name1;char sex;float score; ;class Graduate:pub
28、lic Teacher,public Student public:Graduate(string nam,int a,char s, string t,float sco,float w): Teacher(nam,a,t),Student(nam,s,sco),wage(w) void show( ) coutname:nameendl; coutage:ageendl; coutsex:sexendl; coutscore:scoreendl; couttitle:titleendl; coutwages:wageendl; -private: float wage; ;int main
29、( )Graduate grad1(Wang-li,24,m,assistant,89.5,1234.5);grad1.show( );return 0;在兩個基類中分別用name和name1來代表姓名,其實這是同一個人的名字,從Graduate類的構(gòu)造函數(shù)中可以看到總參數(shù)表中的參數(shù)nam分別傳遞給兩個基類的構(gòu)造函數(shù),作為基類構(gòu)造函數(shù)的實參。-解決這個問題有一個好方法: 在兩個基類中可以都使用同一個數(shù)據(jù)成員名name,而在show函數(shù)中引用數(shù)據(jù)成員時指明其作用域,如coutname:Teacher:nameendl;這就是惟一的,不致引起二義性,能通過編譯,正常運行。通過這個程序還可以發(fā)現(xiàn)一個
30、問題: 在多重繼承時,從不同的基類中會繼承一些重復(fù)的數(shù)據(jù)。如果有多個基類,問題會更突出。在設(shè)計派生類時要細(xì)致考慮其數(shù)據(jù)成員,盡量減少數(shù)據(jù)冗余。4.6.3 多重繼承引起的二義性問題-多重繼承可以反映現(xiàn)實生活中的情況,能夠有效地處理一些較復(fù)雜的問題,使編寫程序具有靈活性,但是多重繼承也引起了一些值得注意的問題,它增加了程序的復(fù)雜度,使程序的編寫和維護變得相對困難,容易出錯。其中最常見的問題就是繼承的成員同名而產(chǎn)生的二義性(ambiguous)問題。a)多個基類中有同名成員-#include #include using namespace std;class Aprotected:int a;pu
31、blic:A(int a)this-a=a;void display( )coutA:a=aa=a;void display( )coutB:a=ab=b;void show()A:display(); B:display();coutC:b=bendl;void main()C c(1,2,3);/c.display(); 二義性;c.show();c.B:display();-所以類中數(shù)據(jù)成員全名應(yīng)該為下圖所示:-b)多個基類和派生類中都有同名成員class C :public A,public Bprivate: int a;public:C(int Aa,int Ba,int Ca):
32、A(Aa),B(Ba)a=Ca;void display()A:display();B:display();coutC:a=aendl;將前面的類改為如下的形式:-這時類的成員構(gòu)成如左圖所示;在類中有三個a,三個display()函數(shù)。思考:如下的main函數(shù)能否執(zhí)行:void main()C c(1,2,3);c.display();c.A:display();c.B:display();這時c.dispaly是可以執(zhí)行的。原因是類中提供的display函數(shù)隱藏了基類和基類中的display函數(shù)。所以直接訪問display函數(shù)是在訪問類中新增加的成員函數(shù)display();-C)如果類A和類
33、B是從同一個基類派生的-前面提到派生類的對象也是基類的對象,因為派生類中繼承了基類中的所有成員(除構(gòu)造函數(shù)和析構(gòu)函數(shù))。準(zhǔn)確的說應(yīng)該是:公有派生類的對象是基類的對象。因為只有公有派生類中成員的訪問屬性與基類完全相同?;惸軐崿F(xiàn)的功能在公有派生類中一定能夠?qū)崿F(xiàn)。4.7 基類與派生類的轉(zhuǎn)換基本數(shù)據(jù)類型在一定條件下可以進(jìn)行類型轉(zhuǎn)化,那么基類對象與派生類對象之間是不是也可以進(jìn)行轉(zhuǎn)化?由于公有派生類對象也是基類的對象,所以派生類對象可以自動轉(zhuǎn)化為基類對象。表現(xiàn)在以下幾方面:-#include #include using namespace std;class Personpublic:Person(s
34、tring nam,char s,int a)name=nam;sex=s;age=a;Person()diplay()cout姓名:nameendl;cout性別:sexendl;cout年齡:ageendl;protected: string name;char sex;int age;-class Teacher: public Person public: Teacher(string nam,char s,int a, string t):Person(nam,s,a)title=t; Teacher() diplay()Person:diplay();cout職稱:diplay();用派生類對象的地址來初始化基類指針后,只能通過該指針訪問派生類中繼承的基類的成員。不能訪問派生類中增加的成員。-()
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 保密協(xié)議合同的侵權(quán)責(zé)任
- 獨家授權(quán)代理商的合同
- 房屋買賣合同如何處理抵押
- 廉政合同采購過程中的道德規(guī)范
- 建筑材料采購安裝合同范本
- 2024年度文化產(chǎn)業(yè)項目投資顧問聘用合同書3篇
- 2024年度企業(yè)人力資源招聘與配置顧問合同3篇
- 2024-2030年防靜電椅公司技術(shù)改造及擴產(chǎn)項目可行性研究報告
- 2024-2030年腈綸抽絲公司技術(shù)改造及擴產(chǎn)項目可行性研究報告
- 2024-2030年精小型電容搬遷改造項目可行性研究報告
- 20K607 防排煙及暖通防火設(shè)計審查與安裝
- 《金剛石、石墨和C60》第一課時名師課件
- 2024年安徽合肥市建設(shè)工程監(jiān)測中心有限責(zé)任公司招聘筆試參考題庫含答案解析
- 滑雪指導(dǎo)員理論考試復(fù)習(xí)題庫(含答案)
- 兩癌篩查年度工作計劃實施方案
- 2024年常德市高三一模語文試卷(含答案)
- 帶你聽懂中國傳統(tǒng)音樂智慧樹知到期末考試答案2024年
- 南京市秦淮區(qū)2022-2023七年級上學(xué)期期中語文試卷及答案
- 肺癌伴咯血護理查房
- 上海市監(jiān)理通用表
- 學(xué)校歸屬感量表
評論
0/150
提交評論