詳談C++何時(shí)需要定義賦值/復(fù)制構(gòu)造函數(shù)
繼承和動(dòng)態(tài)內(nèi)存分配
假設(shè)基類使用了動(dòng)態(tài)內(nèi)存分配,而且定義了析構(gòu)函數(shù)、復(fù)制構(gòu)造函數(shù)和賦值函數(shù),但是在派生類中沒有使用動(dòng)態(tài)內(nèi)存分配,那么在派生類中不需要顯示定義析構(gòu)函數(shù)、復(fù)制構(gòu)造函數(shù)和賦值函數(shù)。
當(dāng)基類和派生類采用動(dòng)態(tài)內(nèi)存分配時(shí),派生類的析構(gòu)函數(shù)、復(fù)制構(gòu)造函數(shù)、賦值運(yùn)算符都必須使用相應(yīng)的基類方法來處理基類元素。這種要求是通過三種不同的方式來滿足的。對(duì)于析構(gòu)函數(shù)。這是自動(dòng)完成的,也就是說在派生類的析構(gòu)函數(shù)中無需顯示調(diào)用基類的析構(gòu)函數(shù)。對(duì)于構(gòu)造函數(shù),這是通過在初始化成員列表中調(diào)用基類的復(fù)制構(gòu)造函數(shù)來完成的,如果不這樣做,將自動(dòng)調(diào)用基類的默認(rèn)構(gòu)造函數(shù),對(duì)于賦值運(yùn)算符,這是通過使用域解析運(yùn)算符顯示地調(diào)用基類的賦值運(yùn)算符來完成的。
編譯器生成的成員函數(shù)
1、 默認(rèn)構(gòu)造函數(shù)
默認(rèn)構(gòu)造哈數(shù)要么沒有參數(shù),要么所有的參數(shù)都有默認(rèn)值。如果沒有定義任何構(gòu)造函數(shù),編譯器將定義構(gòu)造函數(shù)。另外,如果派生類構(gòu)造函數(shù)的成員初始化列表中沒有顯示調(diào)用基類構(gòu)造函數(shù),則編譯器將使用基類的默認(rèn)構(gòu)造函數(shù)來構(gòu)造派生類對(duì)象的基類部分。在這種情況下,如果基類沒有構(gòu)造函數(shù),將導(dǎo)致編譯階段錯(cuò)誤。如果定義了某種構(gòu)造函數(shù),編譯器將不會(huì)定義默認(rèn)構(gòu)造函數(shù)。在這種情況下,如果需要默認(rèn)構(gòu)造函數(shù),則必須自己提供。(最好的建議就是,一旦自己定義了構(gòu)造函數(shù),那么最好再定義一個(gè)默認(rèn)的構(gòu)造函數(shù),這樣在別的子類中調(diào)用也非常的方便)
提供構(gòu)造函數(shù)的動(dòng)機(jī)之一是確保對(duì)象總能被正確地初始化。另外,如果類包含指針成員,則必須初始化這些成員。因此,最好提供一個(gè)顯示默認(rèn)構(gòu)造函數(shù),將所有的類數(shù)據(jù)成員初始化為合理的值。
2、 復(fù)制構(gòu)造函數(shù)
復(fù)制構(gòu)造函數(shù)接受其所屬類的對(duì)象作為參數(shù)。在下述情況下,將使用復(fù)制構(gòu)造函數(shù)
將新對(duì)象初始化為一個(gè)同類對(duì)象
按值將對(duì)象傳遞給函數(shù)
函數(shù)按值返回對(duì)象
編譯器生成臨時(shí)對(duì)象
如果程序沒有使用復(fù)制構(gòu)造函數(shù),編譯器將提供原型,但不提供函數(shù)定義,否則,程序?qū)⒍x一個(gè)執(zhí)行成員初始化的賦值構(gòu)造函數(shù)。也就是說,新對(duì)象的每個(gè)成員都被初始化為原始對(duì)象相應(yīng)成員的值。如果成員為類對(duì)象,則初始化該成員時(shí),將使用相應(yīng)類的復(fù)制構(gòu)造函數(shù)。
如果使用new初始化的成員指針通常要求執(zhí)行深度復(fù)制,或者類可能包含需要修改的靜態(tài)變量。在上述情況下,需要定義自己的復(fù)制構(gòu)造函數(shù)。
3、 賦值構(gòu)造函數(shù)
默認(rèn)的賦值運(yùn)算符用于處理同類對(duì)象之間的賦值。不要將賦值與初始化混淆了。如果語(yǔ)句創(chuàng)建新的對(duì)象,則使用初始化,如果語(yǔ)句修改已有對(duì)象的值,則是賦值。
默認(rèn)賦值為成員賦值。如果成員為類對(duì)象,則默認(rèn)成員賦值將使用相應(yīng)類的賦值運(yùn)算符。如果需要顯示定義復(fù)制構(gòu)造函數(shù),則基于相同的原因。也需要顯示定義賦值運(yùn)算符。
對(duì)于派生類而言,保護(hù)成員類似于共有成員,但對(duì)于外部而言,保護(hù)成員于私有成員類似。派生類可以直接訪問基類的保護(hù)成員,但只能通過基類的成員函數(shù)來訪問私有成員。
基類的析構(gòu)函數(shù)應(yīng)當(dāng)是虛的。這樣,當(dāng)通過指向?qū)ο蟮幕愔羔樆蛞脕韯h除派生對(duì)象時(shí),程序?qū)⑹紫日{(diào)用派生類的析構(gòu)函數(shù),然后調(diào)用基類的析構(gòu)函數(shù),而不僅僅是調(diào)用基類的析構(gòu)函數(shù)
如何判斷一個(gè)指針指向的對(duì)象的真實(shí)類型,使用C++中的運(yùn)行時(shí)機(jī)制,typeid就可以實(shí)現(xiàn)目標(biāo)
以上就是小編為大家?guī)淼脑斦凜++何時(shí)需要定義賦值/復(fù)制構(gòu)造函數(shù)全部?jī)?nèi)容了,希望大家多多支持腳本之家~
相關(guān)文章
C語(yǔ)言中#define與typedef的互換細(xì)節(jié)詳解
本篇文章是對(duì)C語(yǔ)言中#define與typedef的互換細(xì)節(jié)進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05c++中的volatile和variant關(guān)鍵字詳解
大家好,本篇文章主要講的是c++中的volatile和variant關(guān)鍵字詳解,感興趣的同學(xué)趕快來看一看吧,對(duì)你有幫助的話記得收藏一下2022-01-01Qt掃盲篇之QRegularExpression正則匹配總結(jié)
QRegularExpression是Qt5.0引進(jìn)的,修復(fù)了很多bug,提高了效率,使用時(shí)建議使用QRegularExpression,下面這篇文章主要給大家介紹了關(guān)于Qt掃盲篇之QRegularExpression正則匹配的相關(guān)資料,需要的朋友可以參考下2023-03-03C++實(shí)現(xiàn)LeetCode(兩個(gè)有序數(shù)組的中位數(shù))
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(兩個(gè)有序數(shù)組的中位數(shù)),本篇文章通過簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07VsCode搭建C語(yǔ)言運(yùn)行環(huán)境詳細(xì)過程及終端亂碼問題解決方案
這篇文章主要介紹了VsCode搭建C語(yǔ)言運(yùn)行環(huán)境以及終端亂碼問題解決,在VsCode中搭建C/C++運(yùn)行環(huán)境需要先安裝幾個(gè)插件,具體插件文中給大家詳細(xì)介紹,需要的朋友可以參考下2022-12-12利用C++ R3層斷鏈實(shí)現(xiàn)模塊隱藏功能
在R3層的模塊隱藏,我們需要做的就是將其該鏈表斷鏈,將某一模塊從這個(gè)雙向鏈表中摘除,這樣再調(diào)用傳統(tǒng)的API時(shí)就會(huì)搜索不到。本文重點(diǎn)給大家介紹利用C++ R3層斷鏈實(shí)現(xiàn)模塊隱藏功能,感興趣的朋友一起看看吧2019-10-10