【第15章】多重繼承

1. 虛基類介紹

多繼承時很容易產生命名沖突,即使我們很小心地將所有類中的成員變量和成員函數都命名為不同的名字,命名沖突依然有可能發生,比如非常經典的菱形繼承層次。如下圖所示:

類A派生出類B和類C,類D繼承自類B和類C,這個時候類A中的成員變量和成員函數繼承到類D中變成了兩份,一份來自 A-->B-->D 這一路,另一份來自 A-->C-->D 這一條路。

在一個派生類中保留間接基類的多份同名成員,雖然可以在不同的成員變量中分別存放不同的數據,但大多數情況下這是多余的:因為保留多份成員變量不僅占用較多的存儲空間,還容易產生命名沖突,而且很少有這樣的需求。

為了解決這個問題,C++提供了虛基類,使得在派生類中只保留間接基類的一份成員。

聲明虛基類只需要在繼承方式前面加上?virtual?關鍵字,請看下面的例子:

#include <iostream>
using namespace std;class A {
protected:int a;
public:A(int a) : a(a) {}
};class B : virtual public A {
protected:int b;
public:B(int a, int b) : A(a), b(b) {}
};class C : virtual public A {
protected:int c;
public:C(int a, int c) : A(a), c(c) {}
};class D : virtual public B, virtual public C {
private:int d;
public:D(int a, int b, int c, int d) : A(a), B(a, b), C(a, c), d(d) {}void display();
};void D::display()
{cout << "a = " << a << endl;cout << "b = " << b << endl;cout << "c = " << c << endl;cout << "d = " << d << endl;
}int main()
{(new D(1, 2, 3, 4))->display();return 0;
}

輸出結果:

本例中我們使用了虛基類,在派生類D中只有一份成員變量 a 的拷貝,所以在 display() 函數中可以直接訪問 a,而不用加類名和域解析符。

請注意派生類D的構造函數,與以往的用法有所不同。以往,在派生類的構造函數中只需負責對其直接基類初始化,再由其直接基類負責對間接基類初始化。現在,由于虛基類在派生類中只有一份成員變量,所以對這份成員變量的初始化必須由派生類直接給出。如果不由最后的派生類直接對虛基類初始化,而由虛基類的直接派生類(如類B和類C)對虛基類初始化,就有可能由于在類B和類C的構造函數中對虛基類給出不同的初始化參數而產生矛盾。所以規定:在最后的派生類中不僅要負責對其直接基類進行初始化,還要負責對虛基類初始化。

有的讀者會提出:類D的構造函數通過初始化表調了虛基類的構造函數A,而類B和類C的構造函數也通過初始化表調用了虛基類的構造函數A,這樣虛基類的構造函數豈非被調用了3次?大家不必過慮,C++編譯系統只執行最后的派生類對虛基類的構造函數的調用,而忽略虛基類的其他派生類(如類B和類C)對虛基類的構造函數的調用,這就保證了虛基類的數據成員不會被多次初始化。

最后請注意:為了保證虛基類在派生類中只繼承一次,應當在該基類的所有直接派生類中聲明為虛基類,否則仍然會出現對基類的多次繼承。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/385460.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/385460.shtml
英文地址,請注明出處:http://en.pswp.cn/news/385460.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

gcc編譯器與g++編譯器的區別

gcc與g編譯器的程序文件分別為&#xff1a;/usr/bin/g和/usr/bin/gcc。 gcc 和 GCC 是兩個不同的東西&#xff0c;GCC:GNU Compiler Collection(GUN 編譯器集合)&#xff0c;它可以編譯C、C、JAV、Fortran、Pascal、Object-C、Ada等語言。gcc是GCC中的GUN C Compiler&#xff0…

1. 排序算法

一、概述 假定在待排序的記錄序列中&#xff0c;存在多個具有相同的關鍵字的記錄&#xff0c;若經過排序&#xff0c;這些記錄的相對次序保持不變&#xff0c;即在原序列中&#xff0c;r[i]r[j]&#xff0c;且r[i]在r[j]之前&#xff0c;而在排序后的序列中&#xff0c;r[i]仍…

1036. 跟奧巴馬一起編程(15)

美國總統奧巴馬不僅呼吁所有人都學習編程&#xff0c;甚至以身作則編寫代碼&#xff0c;成為美國歷史上首位編寫計算機代碼的總統。2014年底&#xff0c;為慶祝“計算機科學教育周”正式啟動&#xff0c;奧巴馬編寫了很簡單的計算機代碼&#xff1a;在屏幕上畫一個正方形。現在…

庫文件與頭文件

首先說明庫文件與頭文件在gcc中的具體使用方法&#xff0c;然后說明兩者的區別與聯系。 庫文件即庫函數&#xff0c;如printf和scanf函數。以libgtdf.so庫文件為例&#xff08;庫文件在命名時都以lib開頭&#xff0c;因此使用-l選項去鏈接指定的庫文件時可以省略lib三個字母&am…

gcc的常用參數

-c 編譯成目標文件.o&#xff08;只編譯不鏈接&#xff09; gcc -c hello.s -o hello.o -o 指出輸出文件名&#xff0c;輸出文件名跟在-o后面。如果不使用這一選項&#xff0c;則缺省的輸出文件名為a.out。gcc hello.c -o hello.exe&#xff08;在Linux中該項后綴名無要求&a…

1027. 打印沙漏(20)

本題要求你寫個程序把給定的符號打印成沙漏的形狀。例如給定17個“*”&#xff0c;要求按下列格式打印 ************ *****所謂“沙漏形狀”&#xff0c;是指每行輸出奇數個符號&#xff1b;各行符號中心對齊&#xff1b;相鄰兩行符號數差2&#xff1b;符號數先從大到小順序遞減…

【C++ Priemr | 15】構造函數與拷貝控制

繼承的構造函數 1. 簡介&#xff1a; 子類為完成基類初始化&#xff0c;在C11之前&#xff0c;需要在初始化列表調用基類的構造函數&#xff0c;從而完成構造函數的傳遞。如果基類擁有多個構造函數&#xff0c;那么子類也需要實現多個與基類構造函數對應的構造函數。 class …

C命令行參數

C命令行參數的作用是在執行程序時&#xff0c;可以將命令行的參數傳值給C程序內部&#xff0c;這樣就可以從外部控制程序&#xff0c;而不是在代碼內對這些值進行硬編碼。命令行參數是使用main函數來處理的&#xff0c;argc是指參數的個數&#xff0c;為int類型&#xff1b;arg…

剖析數組名、函數名(不是指針常量,更不是指針)

對于一個數組&#xff0c;如 int a[4]; 如果只是給出數組名a&#xff0c;編譯器不知道該取該數組的第幾個元素&#xff0c;因此編譯器不會自動取值&#xff0c;而是返回該數組的首地址&#xff08;第一個元素的地址&#xff09;。其實&#xff0c;數組名a就是數組本身&#xf…

【C++ Priemr | 15】面向對象程序設計

類型準換與繼承 為了支持c的多態性&#xff0c;才用了動態綁定和靜態綁定。 需要理解四個名詞&#xff1a; 對象的靜態類型&#xff1a;對象在聲明時采用的類型&#xff0c;是在編譯期確定的。對象的動態類型&#xff1a;目前所指對象的類型&#xff0c;是在運行期決定的。對…

linux里source、. 、sh、bash、./有什么區別

轉載&#xff1a;https://www.cnblogs.com/pcat/p/5467188.html 1.source a.sh source可以簡寫為“.”&#xff0c;即. a.sh 注意中間有空格&#xff0c;在當前shell內去讀取、執行a.sh&#xff0c;而a.sh不需要有"執行權限"。 2.sh a.sh 和 bash a.sh 都是打開…

【C++ Priemr | 15】虛函數表剖析(三)

一、虛擬菱形繼承 #include <iostream> using namespace std;class B { public:int _b; };class C1 :virtual public B { public:int _c1; };class C2 :virtual public B { public:int _c2; };class D :public C1, public C2 { public:int _d; };int main() {cout <&…

gcc的警告提示信息

gcc包含完整的出錯檢查和警告提示功能。采用-pedantic選項&#xff0c;對于不符合ANSI/ISO標準的源代碼會產生相應的警告信息。如&#xff1a;gcc -pedantic hello.c -o hello (main函數返回類型為int&#xff0c;且函數體內要有return 語句&#xff0c;一般為 return 0;) -pe…

1037. 在霍格沃茨找零錢(20)

如果你是哈利波特迷&#xff0c;你會知道魔法世界有它自己的貨幣系統 —— 就如海格告訴哈利的&#xff1a;“十七個銀西可(Sickle)兌一個加隆(Galleon)&#xff0c;二十九個納特(Knut)兌一個西可&#xff0c;很容易。”現在&#xff0c;給定哈利應付的價錢P和他實付的錢A&…

【Leetcode | 6】136. 只出現一次的數字

給定一個非空整數數組&#xff0c;除了某個元素只出現一次以外&#xff0c;其余每個元素均出現兩次。找出那個只出現了一次的元素。 說明&#xff1a; 你的算法應該具有線性時間復雜度。 你可以不使用額外空間來實現嗎&#xff1f; 示例 1: 輸入: [2,2,1] 輸出: 1 示例 2: 輸入…

gcc的優化功能

代碼優化的目的是改善程序的執行性能。gcc提供的代碼優化功能非常強大&#xff0c;它通過參數-On來控制優化代碼的生成&#xff0c;其中n為優化級別的整數&#xff0c;比較典型的范圍是從0變化到2或3&#xff08;與版本有關&#xff09;。 編譯時通過使用選項-O可以告訴gcc同時…

gcc編譯多個源代碼文件的過程(引出makefile)

由foo1.c foo2.c foo3.c 3個源文件組成的源程序生成最終的可執行程序foo的命令&#xff1a; gcc foo1.c foo2.c foo3.c -o foo 如果處理的源文件不止一個&#xff0c;則gcc會依次對每個文件進行預處理、編譯、匯編&#xff0c;最后將所有的目標代碼和庫文件進行&#xff0c;鏈…

觀擦者模式

/********************************************************************created: 2006/07/20filename: Observer.hauthor: 李創http://www.cppblog.com/converse/purpose: Observer模式的演示代碼 *********************************************************************/…

程序的裝入和鏈接

注&#xff1a;這是本人學習湯小丹等編寫的計算機操作系統&#xff08;西安電子科技大學出版社&#xff09;的學習筆記&#xff0c;因此許多引用來源于此書&#xff0c;在正文中就不注明了&#xff01; 程序在運行前需要經過以下步驟&#xff1a;編譯程序對源程序進行編譯生成…

內存對齊

1. 對齊原則&#xff1a; 數據成員對齊規則&#xff1a;結構(struct)(或聯合(union))的數據成員&#xff0c;第一個數據成員放在offset為0的地方&#xff0c;以后每個數據成員的對齊按照#pragma pack指定的數值和這個數據成員自身長度中&#xff0c;比較小的那個進行。結構(或…