解析C++中不能重載為友元函數(shù)的四個(gè)運(yùn)算符
C++規(guī)定有四個(gè)運(yùn)算符 =, ->, [], ()不可以是全局域中的重載(即不能重載為友員函數(shù)),這是為什么呢?
現(xiàn)在先說說賦值運(yùn)算符“=”的重載
C++規(guī)定賦值運(yùn)算符“=”只能重載為類的非靜態(tài)成員函數(shù),而不可以重載為類的友元函數(shù)。
不能重載為類的靜態(tài)成員應(yīng)該比較容易理解,因?yàn)殪o態(tài)成員函數(shù)是屬于整個(gè)類的,不是屬于某個(gè)對(duì)象的,它只能去操作類靜態(tài)數(shù)據(jù)成員。而賦值運(yùn)算符“=”是基于對(duì)象操作的。
那么為什么賦值運(yùn)算符不可以重載為類的友元函數(shù)?像同樣都是雙目運(yùn)算符的+為什么它就可以呢?
在討論這問題之前,先看一測(cè)試的程序:
#include <iostream>
using namespace std;
class A
{
private:
int x;
public:
A(){x=99;}
A(int xx)
{
cout<<"Call A(int xx)"<<endl;
x = xx;
}
};
int main()
{
A a;
a = 7;
}
程序執(zhí)行結(jié)果為:
Call A(int xx)
說明執(zhí)行a = 7這程序語句時(shí),程序去調(diào)用類A中的帶參構(gòu)造函數(shù)。
在類A中加入一賦值運(yùn)算重載成員函數(shù),如下:
#include <iostream>
using namespace std;
class A
{
private:
int x;
public:
A(){x=99;}
A(int xx)
{
cout<<"Call A(int xx)"<<endl;
x = xx;
}
A operator=(int xx) //重載賦值運(yùn)算符運(yùn)算
{
cout<<"Call A operator=(int xx)"<<endl;
x = xx;
return *this;
}
};
int main()
{
A a;
a = 7;
}
程序運(yùn)行結(jié)果:
Call A operator=(int xx)
說明在類A中已經(jīng)有相應(yīng)賦值運(yùn)算符重載函數(shù)的時(shí)候,執(zhí)行賦值語句a = 7;程序會(huì)去調(diào)用類A中相應(yīng)的賦值運(yùn)算符重載函數(shù),而不會(huì)像上面原來那樣去調(diào)用有參構(gòu)造函數(shù)。
在此,我們可以對(duì)C++規(guī)則做出以下的判斷:
當(dāng)類中沒有定義賦值運(yùn)算符重載成員函數(shù)時(shí)(注意,在未定義形參數(shù)據(jù)類型為該類類型的賦值運(yùn)算符重載函數(shù)時(shí),編譯器會(huì)自動(dòng)生成加入),當(dāng)程序執(zhí)行到某一賦值語句時(shí),程序就會(huì)調(diào)用與賦值語句中右值類型匹配的構(gòu)造函數(shù),而把這右值當(dāng)作此構(gòu)造函數(shù)的實(shí)參。像最初的賦值語句a = 7,執(zhí)行時(shí),實(shí)際做的操作是a(7)。而當(dāng)類中有定義賦值運(yùn)算符重載成員函數(shù),執(zhí)行賦值語句時(shí),程序就只會(huì)去調(diào)用相應(yīng)的賦值運(yùn)算符重載函數(shù)。
當(dāng)明白上面的規(guī)則后,現(xiàn)在就可以回過來,討論為什么賦值運(yùn)算符不可以重載為類的友元函數(shù)了。
我們知道友元函數(shù)不是類的成員函數(shù),它只是類的“朋友“,具有訪問把它聲明為“朋友”的類的數(shù)據(jù)成員的權(quán)限而已。
那么當(dāng)把賦值運(yùn)算符重載為類的友員函數(shù),在程序中執(zhí)行類對(duì)象的賦值語句時(shí),程序就會(huì)出現(xiàn)兩種矛盾的選擇。
1、因?yàn)樗J(rèn)為類中并沒有重載賦值運(yùn)算符的成員函數(shù),所以它根據(jù)C++的規(guī)則,會(huì)去調(diào)用相應(yīng)的構(gòu)造函數(shù)。
2、但是在全局里,我們已經(jīng)重載了參數(shù)類型為此類類型的賦值運(yùn)算符函數(shù),而這賦值語句剛好和這函數(shù)匹配上了,根據(jù)C++的規(guī)則,也會(huì)去調(diào)用這函數(shù)。
程序是不允許有矛盾不確定選擇的,所以當(dāng)賦值運(yùn)算符重載為類的友元函數(shù)時(shí),編譯器就會(huì)提示錯(cuò)誤。
對(duì)于剩下的3個(gè)運(yùn)算符 ->, [], () 為什么不能重載為友元函數(shù),也是跟上面一樣的道理。即編譯器發(fā)現(xiàn)當(dāng)類中沒有定義這3個(gè)運(yùn)算符的重載成員函數(shù)時(shí),就會(huì)自己加入默認(rèn)的運(yùn)算符重載成員函數(shù)。
例當(dāng)類A中沒有定義運(yùn)算符->的重載成員函數(shù),但是我們?nèi)匀豢梢詫?duì)類A對(duì)象指針用->的形式調(diào)用指針指向的對(duì)象里的成員。像類A里有成員函數(shù)f(),當(dāng)
A a;
A* p = &a;
p->f(); //雖然類A中沒有自己定義運(yùn)算符->重載成員函數(shù),但這里仍可這樣使用
然而,當(dāng)我們把->運(yùn)算符重載為類A的友元函數(shù)時(shí),程序就會(huì)出現(xiàn)跟把賦值運(yùn)算符重載友元一樣的情況,即產(chǎn)生矛盾性。
聲明:以上僅為個(gè)人見解
欄 目:C語言
下一篇:C數(shù)據(jù)結(jié)構(gòu)之雙鏈表詳細(xì)示例分析
本文標(biāo)題:解析C++中不能重載為友元函數(shù)的四個(gè)運(yùn)算符
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/4247.html
您可能感興趣的文章
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言中對(duì)數(shù)函數(shù)的表達(dá)式 c語言中對(duì)數(shù)怎么表達(dá)
- 04-02c語言沒有round函數(shù) round c語言
- 04-02C語言中怎么打出三角函數(shù) c語言中怎么打出三角函數(shù)的值
- 01-10數(shù)據(jù)結(jié)構(gòu)課程設(shè)計(jì)- 解析最少換車次數(shù)的問題詳解
- 01-10深入理解C++中常見的關(guān)鍵字含義
- 01-10使用C++實(shí)現(xiàn)全排列算法的方法詳解
- 01-10深入Main函數(shù)中的參數(shù)argc,argv的使用詳解
- 01-10APUE筆記之:進(jìn)程環(huán)境詳解
- 01-10深入解析最長公共子串


閱讀排行
- 1C語言 while語句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹的示例代碼(圣誕
- 3利用C語言實(shí)現(xiàn)“百馬百擔(dān)”問題方法
- 4C語言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對(duì)數(shù)函數(shù)的表達(dá)式 c語言中對(duì)
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機(jī)閱讀
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-10delphi制作wav文件的方法
- 04-02jquery與jsp,用jquery
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 01-10C#中split用法實(shí)例總結(jié)
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什