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

繼承的構造函數

1. 簡介:

子類為完成基類初始化,在C++11之前,需要在初始化列表調用基類的構造函數,從而完成構造函數的傳遞。如果基類擁有多個構造函數,那么子類也需要實現多個與基類構造函數對應的構造函數。

class Base
{
public:Base(int va) : m_value(va), m_c(‘0’) {}Base(char c) : m_c(c), m_value(0) {}
private:int m_value;char m_c;
};class Derived : public Base
{
public:// 初始化基類需要透傳基類的各個構造函數,那么這是很麻煩的Derived(int va) : Base(va) {}Derived(char c) : Base(c) {}void display() { /* dosomething */ } //假設派生類只是添加了一個普通的函數
};

書寫多個派生類構造函數只為傳遞參數完成基類的初始化,這種方式無疑給開發人員帶來麻煩,降低了編碼效率。從C++11開始,推出了繼承構造函數(Inheriting Constructor),使用using來聲明繼承基類的構造函數,我們可以這樣書寫。

class Base {
public:Base(int va) : m_value(va), m_c('0') {}Base(char c) : m_c(c), m_value(0) {}
private:int m_value;char m_c;
};class Derived : public Base {
public: using Base::Base; //使用繼承構造函數    void display() { /* dosomething */ } //假設派生類只是添加了一個普通的函數
};

上面代碼中,我們通過using Base::Base把基類構造函數繼承到派生類中,不再需要書寫多個派生類構造函數來完成基類的初始化。更為巧妙的是,C++11標準規定,繼承構造函數與類的一些默認函數(默認構造、析構、拷貝構造函數等)一樣,是隱式聲明,如果一個繼承構造函數不被相關代碼使用,編譯器不會為其產生真正的函數代碼。這樣比通過派生類構造函數“透傳構造函數參數”來完成基類初始化的方案,總是需要定義派生類的各種構造函數更加節省目標代碼空間。

?

2. 注意事項

(1)繼承構造函數無法初始化派生類數據成員。

繼承構造函數的功能是初始化基類,對于派生類數據成員的初始化則無能為力。解決的辦法主要有兩個:一是使用C++11特性就地初始化成員變量,可以通過=、{}對非靜態成員快速地就地初始化,以減少多個構造函數重復初始化變量的工作,注意初始化列表會覆蓋就地初始化操作。

class Derived :public Base
{
public:using Base::Base; //使用繼承構造函數    void display() { /* dosomethin */ } //假設派生類只是添加了一個普通的函數
private:double m_double{ 0.0 }; //派生類新增數據成員
};

?

(2)構造函數擁有默認值會產生多個構造函數版本,且繼承構造函數無法繼承基類構造函數的默認參數,所以我們在使用有默認參數構造函數的基類時就必須要小心。

class A 
{
public:A(int a = 3, double b = 4) : m_a(a), m_b(b) {}void display() { cout << m_a << " " << m_b << endl; }private:int m_a;double m_b;
};class B : public A 
{
public:using A::A;
};

那么A中的構造函數會有下面幾個版本:

1

2

3

4

A()

A(int)

A(int,double)

A(constA&)

那么B中對應的繼承構造函數將會包含如下幾個版本:?

1

2

3

4

B()

B(int)

B(int,double)

B(constB&)

可以看出,參數默認值會導致多個構造函數版本的產生,因此在使用時需格外小心。

?

(3)多繼承的情況下,繼承構造函數會出現“沖突”的情況,因為多個基類中的部分構造函數可能導致派生類中的繼承構造函數的函數名、參數(即函數簽名)相同。考察如下代碼:

class A {
public:A(int i) {}
};class B {
public:B(int i) {}
};class C : public A, public B {
public:using A::A;using B::B;  //編譯出錯,重復定義C(int)C(int i) :A(i), B(i) {} //顯示定義繼承構造函數C(int)
};

為避免繼承構造函數沖突,可以通過顯示定義繼承類沖突的構造函數,組織隱式生成相應的繼承構造函數。

此外,使用繼承構造函數時,還需要注意以下幾點:

  • 如果基類構造函數被申明為私有成員函數,或者派生類是從基類中虛繼承的 ,那么就不能在派生類中申明繼承構造函數;
  • 一旦使用繼承構造函數,編譯器就不會再為派生類生成默認構造函數了。

?

3. 測試代碼:

#include <iostream>
using namespace std;class Base {
public:Base(int va) : m_value(va), m_c('0') {}Base(char c) : m_c(c), m_value(0) {}int m_value;char m_c;
};class Derived : public Base {
public:using Base::Base;void display() { cout << m_value << endl << m_c << endl; }
};int main()
{Derived d(5);d.display();return 0;
}

輸出結果:

?

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

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

相關文章

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;比較小的那個進行。結構(或…

1006. 換個格式輸出整數 (15)

讓我們用字母B來表示“百”、字母S表示“十”&#xff0c;用“12...n”來表示個位數字n&#xff08;<10&#xff09;&#xff0c;換個格式來輸出任一個不超過3位的正整數。例如234應該被輸出為BBSSS1234&#xff0c;因為它有2個“百”、3個“十”、以及個位的4。 輸入格式&a…

靜態庫的制作和使用

Linux下的靜態庫為lib*.a格式的二進制文件&#xff08;目標文件&#xff09;&#xff0c;對應于Windows下的.lib格式的文件。 &#xff08;1&#xff09;命名規則 lib庫名字 .a libMytest.a &#xff0c;則庫名字為mytest。下面以具體的代碼為例介紹如何制作靜態庫。 //mai…

IO多路復用之select

int select(int nfds, fd_set *readfds, fd_set *writefds,fd_set *exceptfds, struct timeval *timeout); 分析&#xff1a; nfds: 監控的文件描述符集里最大文件描述符加1&#xff0c;因為此參數會告訴內核檢測前多少個文件描述符的狀態 readfds&#xff1a; …

1031. 查驗身份證(15)

一個合法的身份證號碼由17位地區、日期編號和順序編號加1位校驗碼組成。校驗碼的計算規則如下&#xff1a; 首先對前17位數字加權求和&#xff0c;權重分配為&#xff1a;{7&#xff0c;9&#xff0c;10&#xff0c;5&#xff0c;8&#xff0c;4&#xff0c;2&#xff0c;1&…

虛擬地址空間

對于每一個進程都會對應一個虛擬地址空間&#xff0c;對于32位的操作系統&#xff08;其指令的位數最大為32位&#xff0c;因此地址碼最多32位&#xff09;&#xff0c;虛擬地址空間的大小為B即0~4GB的虛擬地址空間&#xff0c;其中內核空間為1GB&#xff0c;如下所示&#xff…

Leecode 69. x 的平方根

實現 int sqrt(int x) 函數。 計算并返回 x 的平方根&#xff0c;其中 x 是非負整數。 由于返回類型是整數&#xff0c;結果只保留整數的部分&#xff0c;小數部分將被舍去。 示例 1: 輸入: 4 輸出: 2 示例 2: 輸入: 8 輸出: 2 說明: 8 的平方根是 2.82842..., 由于返回類…

1002. 寫出這個數 (20)

讀入一個自然數n&#xff0c;計算其各位數字之和&#xff0c;用漢語拼音寫出和的每一位數字。 輸入格式&#xff1a;每個測試輸入包含1個測試用例&#xff0c;即給出自然數n的值。這里保證n小于10100。 輸出格式&#xff1a;在一行內輸出n的各位數字之和的每一位&#xff0c;拼…