解析C++編程中的繼承方面的運(yùn)用
C++繼承與組合詳解
我們知道,在一個(gè)類(lèi)中可以用類(lèi)對(duì)象作為數(shù)據(jù)成員,即子對(duì)象(詳情請(qǐng)查看:C++有子對(duì)象的派生類(lèi)的構(gòu)造函數(shù))。實(shí)際上,對(duì)象成員的類(lèi)型可以是本派生類(lèi)的基類(lèi),也可以是另外一個(gè)已定義的類(lèi)。在一個(gè)類(lèi)中以另一個(gè)類(lèi)的對(duì)象作為數(shù)據(jù)成員的,稱(chēng)為類(lèi)的組合(composition)。
例如,聲明Professor(教授)類(lèi)是Teacher(教師)類(lèi)的派生類(lèi),另有一個(gè)類(lèi)BirthDate(生日),包含year,month,day等數(shù)據(jù)成員??梢詫⒔淌谏盏男畔⒓尤氲絇rofessor類(lèi)的聲明中。如:
class Teacher //教師類(lèi) { public: // Some Code private: int num; string name; char sex; }; class BirthDate //生日類(lèi) { public: // Some Code private: int year; int month; int day; }; class Professor:public Teacher //教授類(lèi) { public: // Some Code private: BirthDate birthday; //BirthDate類(lèi)的對(duì)象作為數(shù)據(jù)成員 };
類(lèi)的組合和繼承一樣,是軟件重用的重要方式。組合和繼承都是有效地利用已有類(lèi)的資源。但二者的概念和用法不同。通過(guò)繼承建立了派生類(lèi)與基類(lèi)的關(guān)系,它是一種 “是”的關(guān)系,如“白貓是貓”,“黑人是人”,派生類(lèi)是基類(lèi)的具體化實(shí)現(xiàn),是基類(lèi)中的一 種。通過(guò)組合建立了成員類(lèi)與組合類(lèi)(或稱(chēng)復(fù)合類(lèi))的關(guān)系,在本例中BirthDate是成員類(lèi),Professor是組合類(lèi)(在一個(gè)類(lèi)中又包含另一個(gè)類(lèi)的對(duì)象成員)。它們之間不是‘‘是”的 關(guān)系,而是“有”的關(guān)系。不能說(shuō)教授(Professor)是一個(gè)生日(BirthDate),只能說(shuō)教授(Professor)有一個(gè)生日(BirthDate)的屬性。
Professor類(lèi)通過(guò)繼承,從Teacher類(lèi)得到了num,name,age,sex等數(shù)據(jù)成員,通過(guò)組合,從BirthDate類(lèi)得到了year,month,day等數(shù)據(jù)成員。繼承是縱向的,組合是橫向的。
如果定義了Professor對(duì)象prof1,顯然prof1包含了生日的信息。通過(guò)這種方法有效地組織和利用現(xiàn)有的類(lèi),大大減少了工作量。如果有
void fun1(Teacher &); void fun2(BirthDate &);
在main函數(shù)中調(diào)用這兩個(gè)函數(shù):
fun1(prof1); //正確,形參為T(mén)eacher類(lèi)對(duì)象的引用,實(shí)參為T(mén)eacher類(lèi)的子類(lèi)對(duì)象,與之賦值兼容 fun2(prof1.birthday); //正確,實(shí)參與形參類(lèi)型相同,都是BirthDate類(lèi)對(duì)象 fun2(prof1); //錯(cuò)誤,形參要求是BirthDate類(lèi)對(duì)象,而prof1是Professor類(lèi)型,不匹配
如果修改了成員類(lèi)的部分內(nèi)容,只要成員類(lèi)的公用接口(如頭文件名)不變,如無(wú)必要,組合類(lèi)可以不修改。但組合類(lèi)需要重新編譯。
繼承在軟件開(kāi)發(fā)中的重要意義
繼承是面向?qū)ο蠹夹g(shù)的重要內(nèi)容,有了繼承,使軟件的重用成為可能。
過(guò)去,軟件人員開(kāi)發(fā)新的軟件,能從已有的軟件中直接選用完全符合要求的部件不 多,一般都要進(jìn)行許多修改才能使用,實(shí)際上有相當(dāng)部分要重新編寫(xiě),工作童很大??s短軟件開(kāi)發(fā)過(guò)程的關(guān)鍵是鼓勵(lì)軟件重用。繼承機(jī)制解決了這個(gè)問(wèn)題。編寫(xiě)面向?qū)ο蟮某绦驎r(shí)要把注意力放在實(shí)現(xiàn)對(duì)自己有用的類(lèi)上面,對(duì)已有的類(lèi)加以整理和分類(lèi),進(jìn)行剪裁和修改,在此基礎(chǔ)上集中精力編寫(xiě)派生類(lèi)新增加的部分,使這些類(lèi)能夠被程序設(shè)計(jì)的許多領(lǐng)域使用。繼承是C++和C的蟑重要的區(qū)別之一。
由于C++提供了繼承的機(jī)制,這就吸引了許多廠商開(kāi)發(fā)各類(lèi)實(shí)用的類(lèi)庫(kù)。用戶(hù)將它們作為基類(lèi)去建立適合于自己的類(lèi)(即派生類(lèi)),并在此基礎(chǔ)上設(shè)計(jì)自己的應(yīng)用程序。類(lèi)庫(kù)的出現(xiàn)使得軟件的重用更加方便,現(xiàn)在有一些類(lèi)庫(kù)是隨著C++編譯系統(tǒng)賣(mài)給用戶(hù)的。讀者不要認(rèn)為類(lèi)庫(kù)是C++編譯系統(tǒng)的一部分。不同的C++編譯系統(tǒng)提供的由不同廠商開(kāi)發(fā)的類(lèi)庫(kù)一般是不同的。在一個(gè)C++編譯系統(tǒng)環(huán)境下利用類(lèi)庫(kù)開(kāi)發(fā)的稈序,在另一種C++編譯系統(tǒng)環(huán)境下可能不能工作,除非把類(lèi)庫(kù)也移植過(guò)去??紤]到廣大用戶(hù)的情況,目前隨C++編譯系統(tǒng)提供的類(lèi)庫(kù)是比較通用的,但它的針對(duì)性和實(shí)用范圍也隨之受到限制。 隨著C ++在全球的迅速推廣,在世界范圍內(nèi)開(kāi)發(fā)用于各個(gè)領(lǐng)域的類(lèi)庫(kù)的工作正日益興旺。
對(duì)類(lèi)庫(kù)中類(lèi)的聲明一般放在頭文件中,類(lèi)的實(shí)現(xiàn)(函數(shù)的定義部分)是單獨(dú)編譯的,以目標(biāo)代碼形式存放在系統(tǒng)某一目錄下。用戶(hù)使用類(lèi)庫(kù)時(shí),不需要了解源代碼,但必須知道頭文件的使用方法和怎樣去連接這些目標(biāo)代碼(在哪個(gè)子目錄下),以便源程序在編譯后與之連接。
由于基類(lèi)是單獨(dú)編譯的,在程序編譯時(shí)只需對(duì)派生類(lèi)新增的功能進(jìn)行編譯,這就大大提高了調(diào)試程序的效率。如果在必要時(shí)修改了基類(lèi),只要基類(lèi)的公用接口不變,派生類(lèi)不必修改,但基類(lèi)需要重新編譯,派生類(lèi)也必須重新編譯,否則不起作用。
那么,人們?yōu)槭裁催@么看重繼承,要求在軟件開(kāi)發(fā)中使用繼承機(jī)制,盡可能地通過(guò)繼承建立一批新的類(lèi)?為什么不是將已有的類(lèi)加以修改,使之滿足自己應(yīng)用的要求呢?
歸納起來(lái),有以下幾個(gè)原因:
有許多基類(lèi)是被程序的其他部分或其他程序使用的,這些程序要求保留原有的 基類(lèi)不受破壞。使用繼承是建立新的數(shù)據(jù)類(lèi)型,它繼承了基類(lèi)的所有特征,但不改變基類(lèi)本身?;?lèi)的名稱(chēng)、構(gòu)成和訪問(wèn)屬性絲毫沒(méi)有改變,不會(huì)影響其他程序的使用。
用戶(hù)往往得不到基類(lèi)的源代碼。如果想修改已有的類(lèi),必須掌握類(lèi)的聲明和類(lèi)的實(shí)現(xiàn)(成員函數(shù)的定義)的源代碼。但是,如果使用類(lèi)庫(kù),用戶(hù)是無(wú)法知道成員函數(shù)的代碼的,因此也就無(wú)法對(duì)基類(lèi)進(jìn)行修改。
在類(lèi)庫(kù)中,一個(gè)基類(lèi)可能已被指定與用戶(hù)所需的多種組件建立了某種關(guān)系,因此 在類(lèi)庫(kù)中的基類(lèi)是不容許修改的(即使用戶(hù)知道了源代碼,也決不允許修改)。
實(shí)際上,許多基類(lèi)并不是從已有的其他程序中選取來(lái)的,而是專(zhuān)門(mén)作為基類(lèi)設(shè)計(jì)的。有些基類(lèi)可能并沒(méi)有什么獨(dú)立的功能,只是一個(gè)框架,或者說(shuō)是抽象類(lèi)。人們根據(jù)需要設(shè)計(jì)了一批能適用于不同用途的通用類(lèi),目的是建立通用的數(shù)據(jù)結(jié)構(gòu),以便用戶(hù)在此基礎(chǔ)上添加各種功能,從而建立各種功能的派生類(lèi)。
在面向?qū)ο蟪绦蛟O(shè)計(jì)中,需要設(shè)計(jì)類(lèi)的層次結(jié)構(gòu),從最初的抽象類(lèi)出發(fā),每一層派生類(lèi)的建立都逐步地向著目標(biāo)的具體實(shí)現(xiàn)前進(jìn),換句話說(shuō),是不斷地從抽象到具體的過(guò) 程。每一層的派生和繼承都需要站在整個(gè)系統(tǒng)的角度統(tǒng)一規(guī)劃,精心組織。
相關(guān)文章
VS Code 中安裝運(yùn)行、編寫(xiě)C語(yǔ)言程序的詳細(xì)教程
這篇文章主要介紹了VS Code 中安裝運(yùn)行、編寫(xiě)C語(yǔ)言程序的詳細(xì)教程,本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-03-03C++20 新特性 協(xié)程 Coroutines(2)
上篇文章簡(jiǎn)單給大介紹了 C++20 特性 協(xié)程 Coroutines co_yield 和 co_return 那么這篇文章繼續(xù)給大家介紹C++20 的新特性協(xié)程 Coroutines co_await,需要的朋友可以參考一下2021-10-10C語(yǔ)言基本排序算法之插入排序與直接選擇排序?qū)崿F(xiàn)方法
這篇文章主要介紹了C語(yǔ)言基本排序算法之插入排序與直接選擇排序?qū)崿F(xiàn)方法,結(jié)合具體實(shí)例形式分析了插入排序與直接選擇排序的定義、使用方法及相關(guān)注意事項(xiàng),需要的朋友可以參考下2017-09-09C語(yǔ)言中main函數(shù)兩個(gè)參數(shù)的作用
這篇文章主要介紹了C語(yǔ)言中main函數(shù)兩個(gè)參數(shù)的作用,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-09-09C++ const限定符以及頂層const和底層const的案例詳解
這篇文章主要介紹了C++ const限定符以及頂層const和底層const的案例詳解,本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-09-09C++實(shí)現(xiàn)基于不相交集合的O(mlgn)復(fù)雜度的kruskal算法
這篇文章主要為大家詳細(xì)介紹了C++如何實(shí)現(xiàn)基于不相交集合的O(mlgn)復(fù)雜度的kruskal算法,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2023-02-02