關(guān)于C++運(yùn)算符重載的一些困惑詳解
一.背景
在復(fù)習(xí)《C++基礎(chǔ)與提高》時(shí),自己實(shí)現(xiàn)運(yùn)算符重載(i++)時(shí),幾次都報(bào)錯(cuò)。其實(shí)還是自己對(duì)運(yùn)算符重載這一部分內(nèi)容理解得不夠透徹,于是再次看了下書上的內(nèi)容,理解算是加深了一些,于是提筆記錄一下。
環(huán)境:win10,QT4.8
二.概述
這部分內(nèi)容主要關(guān)于在重載函數(shù)中,函數(shù)前要不要加const,何時(shí)加const,返回類型要不要加&(引用)修飾,何時(shí)加&(引用)的問題,還有臨時(shí)對(duì)象的問題。關(guān)于為什么要重載,重載的規(guī)則,友元重載、成員重載的區(qū)別之類的知識(shí)點(diǎn),這里就不贅述了。
三.內(nèi)容
以類Complex為例
class Complex { public: Complex(double x = 0, double y = 0) :m_x(x), m_y(y){} void dis() { cout<<"("<<m_x<<", "<<m_y<<")"<<endl; } protected: double m_x; double m_y; };
1.以實(shí)現(xiàn)單目運(yùn)算符prefix++和surfix++為例。
先提這個(gè)例子,一是因?yàn)槲以趶?fù)習(xí)這塊時(shí)遇到了一點(diǎn)問題,二是這個(gè)有點(diǎn)特別,涉及到啞元的問題。
prefixe++
1).考慮基本數(shù)據(jù)類型,以int類型為例,如下的操作都是可以的;
int a = 1; ++a; ++++a;
2).先實(shí)現(xiàn)基本的語義,代碼如下:
Complex Complex::operator++(void) { m_x++; m_y++; return *this; }
3)考慮添加
重載函數(shù)返回的是對(duì)象自身,并且需要修改對(duì)象,我們即可以想到返回的是引用類型。注意,此時(shí)引用指向的對(duì)象在重載函數(shù)調(diào)用時(shí)就已經(jīng)存在了。
4)先運(yùn)行一下,看下是否能編譯通過
++c1;
++++c1;
此時(shí)重載函數(shù)實(shí)現(xiàn)的效果,與基本類型效果一致,符合預(yù)期,此時(shí)就不考慮重載函數(shù)前面是否加const修飾了。
#include <iostream> using namespace std; class Complex { public: Complex(double x = 0, double y =0) :m_x(x), m_y(y){} void dis() { cout<<"("<<m_x<<", "<<m_y<<")"<<endl; } Complex & operator++(void); protected: double m_x; double m_y; }; Complex & Complex::operator++(void) { m_x++; m_y++; return *this; } int main() { double a = 1.0; cout<<++a<<endl; ++++a; cout<<a<<endl; Complex c1(1.0, 2.0); Complex cc = ++c1; cc.dis(); cc = ++++c1; // cc = (c1.operator++()).operator++(); cc.dis(); return 0; }
結(jié)果如下
surfix++
為了區(qū)分prefix++和surfix++兩個(gè)成員函數(shù),須使用啞元進(jìn)行區(qū)分(引入 啞元,增加了入?yún)⒌姆绞剑谡{(diào)用時(shí)不需要添加任何的參數(shù)),其實(shí)類似一個(gè)占位符。
1).考慮基本數(shù)據(jù)類型,以int類型為例,可以進(jìn)行的操作和不可以進(jìn)行的操作
int b = 1; b++; // 支持 b++++; // 不支持
2).先實(shí)現(xiàn)基本的語義,代碼如下
Complex operator++(int) { Complex temp = *this; m_x++; m_y++; return temp; }
3)考慮添加
可以觀察到,重載函數(shù)返回的是一個(gè)臨時(shí)對(duì)象。若是串聯(lián)調(diào)用,這個(gè)臨時(shí)對(duì)象它又會(huì)調(diào)用一次此重載函數(shù)
c1.operator++(0).operator++(0);
調(diào)用完,然后就消失了。
此時(shí)切不可在返回類型中添加&。原因如下:
【不要返回局部對(duì)象的引用或指針】
函數(shù)完成后,它所占用的存儲(chǔ)空間也隨之被釋放掉。因此,函數(shù)終止意味著局部變量的引用將指向不再有效的內(nèi)存區(qū)域。同樣地,函數(shù)終止,局部對(duì)象被釋放,指針將指向一個(gè)不存在的對(duì)象。
4)先運(yùn)行一下,看下是否能編譯通過
我們會(huì)發(fā)現(xiàn),第34行無法通過編譯,但是第42行可以通過編譯。
5)重載的運(yùn)算符是否會(huì)導(dǎo)致表達(dá)式可以被賦值,應(yīng)該以基礎(chǔ)類型為準(zhǔn),如int a, b, c; (a=b)=c;是可以的,而(a+b)=c;是不允許的。返回類型通過加const加以限定來實(shí)現(xiàn)。
為了使自定義類型與基本數(shù)據(jù)類型一致,我們?cè)诜祷仡愋颓懊婕由蟘onst。重載函數(shù)中代碼修改為如下
const Complex operator++(int);
修改之后,我們可以看到,第34行和42行均無法通過編譯,符合預(yù)期。
2.雙目運(yùn)算符+
1)考慮基本類型,以下操作都是支持的
int a1 = 1, a2 = 2, a3 = 3; int m; m = a1+a2; m = a1+(a2+a3); m = (a1+a2)+a3;
2)先重載=,成員函數(shù)如下
Complex & Complex::operator=(const Complex &another) { this->m_x = another.m_x; this->m_y = another.m_y; return *this; }
3)再重載運(yùn)算符+,如下:
因?yàn)椴⑽葱薷膫魅氲膮?shù),所以參數(shù)前加了const
Complex Complex::operator+(const Complex &another) { return Complex(this->m_x + another.m_x, this->m_y + another.m_y); }
4)返回類型是否需要加const呢?
我們?cè)賹?duì)比下表達(dá)式的賦值情況,第49行,對(duì)于基本類型,臨時(shí)對(duì)象被賦值的情況編譯無法通過,但是第58行,自定義類型卻編譯通過了。此時(shí),為了使其編譯不過,可通過在返回值類型前加const加以限定。
將代碼
Complex Complex::operator+(const Complex &another);
修改為如下:
const Complex Complex::operator+(const Complex &another);
5)此時(shí),發(fā)現(xiàn)第49和58行均無法通過編譯,同時(shí)第55行和第57行也編譯不過了。
這個(gè)是為啥呢?
再仔細(xì)看剛修改的代碼和第57行代碼。重載函數(shù)返回類型加了const后,返回的就是const對(duì)象了。第57行代碼,c1 + c2 + c3; c1 + c2返回的是const對(duì)象,而重載函數(shù)是一個(gè)非const函數(shù)。此時(shí),即會(huì)報(bào)錯(cuò)。
在const修飾類一節(jié)中,有學(xué)習(xí)過:如果const構(gòu)成函數(shù)重載,const對(duì)象只能調(diào)用const函數(shù),非const對(duì)象優(yōu)先調(diào)用非const函數(shù)。
調(diào)整,在重載函數(shù)后面添加const,如下:
const Complex Complex::operator+(const Complex &another) const;
四.結(jié)尾
學(xué)無止境,繼續(xù)前行,
參考材料
《C++基礎(chǔ)與提高》 王桂林
《C++ Primer》第5版SB、JL、BE
到此這篇關(guān)于C++運(yùn)算符重載的一些困惑的文章就介紹到這了,更多相關(guān)C++運(yùn)算符重載內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++數(shù)據(jù)結(jié)構(gòu)之并查集詳解
這篇文章主要介紹了C++數(shù)據(jù)結(jié)構(gòu)之并查集詳解,并查集是一種樹型的數(shù)據(jù)結(jié)構(gòu),用于處理一些不相交集合的合并及查詢問題,并查集的思想是用一個(gè)數(shù)組表示了整片森林,需要的朋友可以參考下2023-08-08C語言中 int main(int argc,char *argv[])的兩個(gè)參數(shù)詳解
這篇文章主要介紹了C語言中 int main(int argc,char *argv[])的兩個(gè)參數(shù)詳解的相關(guān)資料,需要的朋友可以參考下2017-03-03OpenCV獲取圖像中直線上的數(shù)據(jù)具體流程
對(duì)圖像進(jìn)行處理時(shí),經(jīng)常會(huì)有這類需求:客戶想要提取出圖像中某條直線或者ROI區(qū)域內(nèi)的感興趣數(shù)據(jù),進(jìn)行重點(diǎn)關(guān)注,怎么操作呢,下面小編通過實(shí)例代碼介紹下OpenCV獲取圖像中直線上的數(shù)據(jù),一起看看吧2021-11-11使用WindowsAPI實(shí)現(xiàn)播放PCM音頻的方法
這篇文章主要介紹了使用WindowsAPI實(shí)現(xiàn)播放PCM音頻的方法,很實(shí)用的一個(gè)功能,需要的朋友可以參考下2014-08-08VC++基于Dx實(shí)現(xiàn)的截圖程序示例代碼
這篇文章主要介紹了VC++基于Dx實(shí)現(xiàn)的截圖程序示例代碼,比較實(shí)用的功能,需要的朋友可以參考下2014-07-07C++ 內(nèi)聯(lián)函數(shù)inline案例詳解
這篇文章主要介紹了C++ 內(nèi)聯(lián)函數(shù)inline案例詳解,本篇文章通過簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-09-09