分享
 
 
 

C++運算符重載轉換運算符

王朝c/c++·作者佚名  2008-06-01
窄屏简体版  字體: |||超大  

為什麽需要轉換運算符? 大家知道對於內置類型的數據我們可以通過強制轉換符的使用來轉換數據,例如(int)2.1f;自定義類也是類型,那麽自定義類的對象在很多情況下也需要支持此操作,C++提供了轉換運算符重載函數,它使得自定義類對象的強轉換成為可能。

轉換運算符的生命方式比較非凡,方法如下:

Operator 類名();

轉換運算符的重載函數是沒有返回類型的,它和類的構造函數,析構函數一樣是不遵循函數有返回類型的規定的,他們都沒有返回值。

下面我看一個例子,看看它是如何工作的:

//例1

//程序作者:管寧

//站點:www.cndev-lab.com

//所有稿件均有版權,如要轉載,請務必聞名出處和作者

#include <iostream>

using namespace std;

class Test

{

public:

Test(int a = 0)

{

cout<<this<<":"<<"載入構造函數!"<<a<<endl;

Test::a = a;

}

Test(Test &temp)

{

cout<<"載入拷貝構造函數!"<<endl;

Test::a = temp.a;

}

~Test()

{

cout<<this<<":"<<"載入析構函數!"<<this->a<<endl;

cin.get();

}

operator int()//轉換運算符

{

cout<<this<<":"<<"載入轉換運算符函數!"<<this->a<<endl;

return Test::a;

}

public:

int a;

};

int main()

{

Test b(99);

cout<<"b的內存地址"<<&b<<endl;

cout<<(int)b<<endl;//強轉換

system("pause");

}

在例子中我們利用轉換運算符將Test類的對象強轉換成了int類型並輸出,註重觀察轉換運算符函數的運行狀態,發現並沒有產生臨時對象,證實了它與普通函數並不相同,雖然它帶有return語句。

在很多情況下,類的強轉換運算符還可以作為類對象加運算重載函數使用,盡管他們的意義並不相同,下面的例子,就是利用轉換運算符,將兩個類對象轉換成int後,相加並創建臨時類對象,後再賦給另一個對象。

代碼如下://例2

//程序作者:管寧

//站點:www.cndev-lab.com

//所有稿件均有版權,如要轉載,請務必聞名出處和作者

#include <iostream>

using namespace std;

class Test

{

public:

Test(int a = 0)

{

cout<<this<<":"<<"載入構造函數!"<<a<<endl;

Test::a = a;

}

Test(Test &temp)

{

cout<<"載入拷貝構造函數!"<<endl;

Test::a = temp.a;

}

~Test()

{

cout<<this<<":"<<"載入析構函數!"<<this->a<<endl;

cin.get();

}

operator int()

{

cout<<this<<":"<<"載入轉換運算符函數的內存地址:"<<this->a<<endl;

return Test::a;

}

public:

int a;

};

int main()

{

Test a(100),b(100),c;

cout<<"a的內存地址"<<&a<<" b的內存地址"<<&b<<endl;

c=Test((int)a+(int)b);//顯示式轉換

//c=a+b;//隱式轉換

cout<<"c的內存地址"<<&c<<endl;

cout<<c.a<<endl;

system("pause");

}

代碼中的c=a+b;屬於隱式轉換,它的實現過程與c=Test((int)a+(int)b);完全相同。

運行結果如下圖示(註重觀察內存地址,觀察構造與析構過程,執行過程圖中有解釋):

當一個類含有轉換運算符重載函數的時候,有時候會破壞C++原有規則,導致運算效率降低,這一點不得不註重。

示例如下:

//例3

//程序作者:管寧

//站點:www.cndev-lab.com

//所有稿件均有版權,如要轉載,請務必聞名出處和作者

#include <iostream>

using namespace std;

class Test

{

public:

Test(int a = 0)

{

cout<<this<<":"<<"載入構造函數!"<<a<<endl;

Test::a = a;

}

Test(Test &temp)

{

cout<<"載入拷貝構造函數!"<<endl;

Test::a = temp.a;

}

~Test()

{

cout<<this<<":"<<"載入析構函數!"<<this->a<<endl;

cin.get();

}

operator int()//轉換運算符,去掉則不會調用

{

cout<<this<<":"<<"載入轉換運算符函數的內存地址:"<<this->a<<endl;

return Test::a;

}

public:

int a;

};

int main()

{

Test b=Test(99);//註重這裏

cout<<"b的內存地址"<<&b<<endl;

cout<<b.a<<endl;

system("pause");

}

按照C++對無名對象的約定,Test b=Test(99);C++是會按照Test b(99);來處理的,可是由於轉換運算符的加入,導致這一規律被破壞,系統會「錯誤的」認為你是要給對象賦值,所以系統首先利用Test(99)創建一個臨時對象用於賦值過程使用,可是恰恰系統又沒有使用自動提供的賦值運算重載函數去處理,因為發現b對象並未構造,轉而又不得不將開始原本用於賦值而創建的臨時對象再次的強轉換為int類型,提供給b對象進行構造,可見中間的創建臨時對象和載入轉換運算符函數的過程完全是多余,讀者對此例要認真解讀,充分理解。

運行結果如下圖所示(運行過程的解釋見圖):

由於類的轉換運算符與類的運算符重載函數,在某些地方上使用的時候,有功能相似的地方,假如兩者都存在於類中,那麽雖然運行結果正確,但其運行過程會出現一些意向不到的步驟,導致程序運行效率降低。

下面的例子就是這個情況,代碼如下://例4

//程序作者:管寧

//站點:www.cndev-lab.com

//所有稿件均有版權,如要轉載,請務必聞名出處和作者

#include <iostream>

using namespace std;

class Test

{

public:

Test(int a = 0)

{

cout<<this<<":"<<"載入構造函數!"<<a<<endl;

Test::a = a;

}

Test(Test &temp)

{

cout<<"載入拷貝構造函數!"<<endl;

Test::a = temp.a;

}

~Test()

{

cout<<this<<":"<<"載入析構函數!"<<this->a<<endl;

cin.get();

}

Test operator +(Test& temp2)

{

cout<<this<<""<<&temp2<<"載入加運算符重載函數!"<<endl;

Test result(this->a+temp2.a);

return result;

}

operator int()

{

cout<<this<<":"<<"載入轉換運算符函數的內存地址:"<<this->a<<endl;

return Test::a;

}

public:

int a;

};

int main()

{

Test a(100),b(100);

cout<<"a的內存地址:"<<&a<<" b的內存地址:"<<&b<<endl;

Test c=a+b;

cout<<"c的內存地址:"<<&c<<endl;

cout<<c.a<<endl;

system("pause");

}

運行過程見下圖。

從圖中我們可以清楚的看到,不必要的運算過程被執行,導致開銷增大,讀者在理解此例的時候要格外小心!

現在總結一下轉換運算符的優點與缺點:

優點:在不提供帶有類對象參數的運算符重載函數的情況下,轉換運算符重載函數可以將類對象轉換成需要的類型,然後進行運算,最後在構造成類對象,這一點和類的運算符重載函數有相同的功效。(例2就是這種情況)

缺點:假如一個類只有轉換運算符重載函數,而沒有真正意義上運算符重載函數,當用轉換運算符重載函數替代運算符重載函數,進行工作的時候,就會讓程序的可讀性降低,歪曲了運算符操作的真正含義。(例2中的c=a+b;//隱式轉換,就是例子,事實上a+b的作用只是對返回的整型數據進行了加運算,而對象賦值的操作是系統隱式的幫大家轉換成了c=Test(a+b)。)

最後我們來說一下,多路徑轉換的多義性問題,多義性問題一直是C++編程中輕易忽視的問題,但它的確是不容小視,當問題隱藏起來的時候你不會發覺,一旦觸發麻煩就來了。

類的轉換構造函數與類的轉換運算符重載函數是互逆的。(例3中的Test(int a = 0)是將int類型的數據轉換構造成Test類對象,而operator int()則是將Test類對象轉換成int類型數據)

但是當他們是出現在兩個不同的類中,對於一個類對象轉換來說,同時擁有兩種近似的轉換途徑的時候,多義性的問題就暴露出來,導致編譯出錯。

下例就是這個狀態:

//程序作者:管寧

//站點:www.cndev-lab.com

//所有稿件均有版權,如要轉載,請務必聞名出處和作者

#include <iostream>

using namespace std;

class B;

class A

{

public:

A(B &);//轉換構造函數,他的作用是用B類對象構造A類對象

void Edita(int temp)

{

A::a=temp;

}

public:

int a;

};

class B

{

public:

B(int a=0)

{

B::a=a;

}

int Ra()

{

return B::a;

}

operator A()//轉換運算符重載函數,他的作用則是將B類對象轉換成A類對象

{

return *this;

}

PRotected:

int a;

};

A::A(B &temp)

{

cout<<this<<""<<&temp<<endl;

A::a=temp.Ra();

}

void tp(A temp)

{

}

int main()

{

B BT(100);

A at=A(bt);

//tp(bt);//錯誤,多義性問題,系統不知道如何選擇,是選擇A(B &)轉化構造好呢?還是選擇B::operator A()進行轉換好呢?

tp(A::A(bt));//顯示的處理可以消除多義性問題

system("pause");

}

代碼中的A at=A(bt);運行正常,因為系統發現對象at還未構造,所以優先選取了A類的轉換構造函數處理了,沒有產生二義性問題。

但是代碼中的tp(bt);編譯錯誤,這是因為函數tp的參數要求的是一個A類對象,而我們給他的則是一個B類對象,而在A類與B類中都有一個類似的操作,可以將B類對象轉換成A類對象,系統不知道是選取A類的轉換構造函數進行構造處理,還是選擇B類中的轉換運算符號重載函數處理,系統拒絕從他們兩個中選一個,所以編譯錯誤。

我們修改tp(bt)為tp(A::A(bt));編譯正常,因為我們顯式的明確的告訴系統應該使用A類的轉換構造函數處理,所以,顯式的告訴計算機應該如何處理數據,通常可以解決多義性問題。

 
 
 
免責聲明:本文為網絡用戶發布,其觀點僅代表作者個人觀點,與本站無關,本站僅提供信息存儲服務。文中陳述內容未經本站證實,其真實性、完整性、及時性本站不作任何保證或承諾,請讀者僅作參考,並請自行核實相關內容。
  為什麽需要轉換運算符?  大家知道對於內置類型的數據我們可以通過強制轉換符的使用來轉換數據,例如(int)2.1f;自定義類也是類型,那麽自定義類的對象在很多情況下也需要支持此操作,C++提供了轉換運算符重載函數,它使得自定義類對象的強轉換成為可能。  轉換運算符的生命方式比較非凡,方法如下:    Operator 類名();   轉換運算符的重載函數是沒有返回類型的,它和類的構造函數,析構函數一樣是不遵循函數有返回類型的規定的,他們都沒有返回值。   下面我看一個例子,看看它是如何工作的: //例1 //程序作者:管寧 //站點:www.cndev-lab.com //所有稿件均有版權,如要轉載,請務必聞名出處和作者 #include <iostream> using namespace std; class Test { public: Test(int a = 0) { cout<<this<<":"<<"載入構造函數!"<<a<<endl; Test::a = a; } Test(Test &temp) { cout<<"載入拷貝構造函數!"<<endl; Test::a = temp.a; } ~Test() { cout<<this<<":"<<"載入析構函數!"<<this->a<<endl; cin.get(); } operator int()//轉換運算符 { cout<<this<<":"<<"載入轉換運算符函數!"<<this->a<<endl; return Test::a; } public: int a; }; int main() { Test b(99); cout<<"b的內存地址"<<&b<<endl; cout<<(int)b<<endl;//強轉換 system("pause"); }   在例子中我們利用轉換運算符將Test類的對象強轉換成了int類型並輸出,註重觀察轉換運算符函數的運行狀態,發現並沒有產生臨時對象,證實了它與普通函數並不相同,雖然它帶有return語句。   在很多情況下,類的強轉換運算符還可以作為類對象加運算重載函數使用,盡管他們的意義並不相同,下面的例子,就是利用轉換運算符,將兩個類對象轉換成int後,相加並創建臨時類對象,後再賦給另一個對象。   代碼如下://例2 //程序作者:管寧 //站點:www.cndev-lab.com //所有稿件均有版權,如要轉載,請務必聞名出處和作者 #include <iostream> using namespace std; class Test { public: Test(int a = 0) { cout<<this<<":"<<"載入構造函數!"<<a<<endl; Test::a = a; } Test(Test &temp) { cout<<"載入拷貝構造函數!"<<endl; Test::a = temp.a; } ~Test() { cout<<this<<":"<<"載入析構函數!"<<this->a<<endl; cin.get(); } operator int() { cout<<this<<":"<<"載入轉換運算符函數的內存地址:"<<this->a<<endl; return Test::a; } public: int a; }; int main() { Test a(100),b(100),c; cout<<"a的內存地址"<<&a<<" b的內存地址"<<&b<<endl; c=Test((int)a+(int)b);//顯示式轉換 //c=a+b;//隱式轉換 cout<<"c的內存地址"<<&c<<endl; cout<<c.a<<endl; system("pause"); }   代碼中的c=a+b;屬於隱式轉換,它的實現過程與c=Test((int)a+(int)b);完全相同。   運行結果如下圖示(註重觀察內存地址,觀察構造與析構過程,執行過程圖中有解釋): [img]http://image.wangchao.net.cn/it/1323423594100.gif[/img]   當一個類含有轉換運算符重載函數的時候,有時候會破壞C++原有規則,導致運算效率降低,這一點不得不註重。   示例如下: //例3 //程序作者:管寧 //站點:www.cndev-lab.com //所有稿件均有版權,如要轉載,請務必聞名出處和作者 #include <iostream> using namespace std; class Test { public: Test(int a = 0) { cout<<this<<":"<<"載入構造函數!"<<a<<endl; Test::a = a; } Test(Test &temp) { cout<<"載入拷貝構造函數!"<<endl; Test::a = temp.a; } ~Test() { cout<<this<<":"<<"載入析構函數!"<<this->a<<endl; cin.get(); } operator int()//轉換運算符,去掉則不會調用 { cout<<this<<":"<<"載入轉換運算符函數的內存地址:"<<this->a<<endl; return Test::a; } public: int a; }; int main() { Test b=Test(99);//註重這裏 cout<<"b的內存地址"<<&b<<endl; cout<<b.a<<endl; system("pause"); }   按照C++對無名對象的約定,Test b=Test(99);C++是會按照Test b(99);來處理的,可是由於轉換運算符的加入,導致這一規律被破壞,系統會「錯誤的」認為你是要給對象賦值,所以系統首先利用Test(99)創建一個臨時對象用於賦值過程使用,可是恰恰系統又沒有使用自動提供的賦值運算重載函數去處理,因為發現b對象並未構造,轉而又不得不將開始原本用於賦值而創建的臨時對象再次的強轉換為int類型,提供給b對象進行構造,可見中間的創建臨時對象和載入轉換運算符函數的過程完全是多余,讀者對此例要認真解讀,充分理解。   運行結果如下圖所示(運行過程的解釋見圖): [img]http://image.wangchao.net.cn/it/1323423594279.gif[/img]   由於類的轉換運算符與類的運算符重載函數,在某些地方上使用的時候,有功能相似的地方,假如兩者都存在於類中,那麽雖然運行結果正確,但其運行過程會出現一些意向不到的步驟,導致程序運行效率降低。   下面的例子就是這個情況,代碼如下://例4 //程序作者:管寧 //站點:www.cndev-lab.com //所有稿件均有版權,如要轉載,請務必聞名出處和作者 #include <iostream> using namespace std; class Test { public: Test(int a = 0) { cout<<this<<":"<<"載入構造函數!"<<a<<endl; Test::a = a; } Test(Test &temp) { cout<<"載入拷貝構造函數!"<<endl; Test::a = temp.a; } ~Test() { cout<<this<<":"<<"載入析構函數!"<<this->a<<endl; cin.get(); } Test operator +(Test& temp2) { cout<<this<<""<<&temp2<<"載入加運算符重載函數!"<<endl; Test result(this->a+temp2.a); return result; } operator int() { cout<<this<<":"<<"載入轉換運算符函數的內存地址:"<<this->a<<endl; return Test::a; } public: int a; }; int main() { Test a(100),b(100); cout<<"a的內存地址:"<<&a<<" b的內存地址:"<<&b<<endl; Test c=a+b; cout<<"c的內存地址:"<<&c<<endl; cout<<c.a<<endl; system("pause"); }   運行過程見下圖。 [img]http://image.wangchao.net.cn/it/1323423594410.gif[/img]   從圖中我們可以清楚的看到,不必要的運算過程被執行,導致開銷增大,讀者在理解此例的時候要格外小心! 現在總結一下轉換運算符的優點與缺點:   優點:在不提供帶有類對象參數的運算符重載函數的情況下,轉換運算符重載函數可以將類對象轉換成需要的類型,然後進行運算,最後在構造成類對象,這一點和類的運算符重載函數有相同的功效。(例2就是這種情況)   缺點:假如一個類只有轉換運算符重載函數,而沒有真正意義上運算符重載函數,當用轉換運算符重載函數替代運算符重載函數,進行工作的時候,就會讓程序的可讀性降低,歪曲了運算符操作的真正含義。(例2中的c=a+b;//隱式轉換,就是例子,事實上a+b的作用只是對返回的整型數據進行了加運算,而對象賦值的操作是系統隱式的幫大家轉換成了c=Test(a+b)。)   最後我們來說一下,多路徑轉換的多義性問題,多義性問題一直是C++編程中輕易忽視的問題,但它的確是不容小視,當問題隱藏起來的時候你不會發覺,一旦觸發麻煩就來了。   類的轉換構造函數與類的轉換運算符重載函數是互逆的。(例3中的Test(int a = 0)是將int類型的數據轉換構造成Test類對象,而operator int()則是將Test類對象轉換成int類型數據) 但是當他們是出現在兩個不同的類中,對於一個類對象轉換來說,同時擁有兩種近似的轉換途徑的時候,多義性的問題就暴露出來,導致編譯出錯。   下例就是這個狀態: //程序作者:管寧 //站點:www.cndev-lab.com //所有稿件均有版權,如要轉載,請務必聞名出處和作者 #include <iostream> using namespace std; class B; class A { public: A(B &);//轉換構造函數,他的作用是用B類對象構造A類對象 void Edita(int temp) { A::a=temp; } public: int a; }; class B { public: B(int a=0) { B::a=a; } int Ra() { return B::a; } operator A()//轉換運算符重載函數,他的作用則是將B類對象轉換成A類對象 { return *this; } PRotected: int a; }; A::A(B &temp) { cout<<this<<""<<&temp<<endl; A::a=temp.Ra(); } void tp(A temp) { } int main() { B BT(100); A at=A(bt); //tp(bt);//錯誤,多義性問題,系統不知道如何選擇,是選擇A(B &)轉化構造好呢?還是選擇B::operator A()進行轉換好呢? tp(A::A(bt));//顯示的處理可以消除多義性問題 system("pause"); }   代碼中的A at=A(bt);運行正常,因為系統發現對象at還未構造,所以優先選取了A類的轉換構造函數處理了,沒有產生二義性問題。   但是代碼中的tp(bt);編譯錯誤,這是因為函數tp的參數要求的是一個A類對象,而我們給他的則是一個B類對象,而在A類與B類中都有一個類似的操作,可以將B類對象轉換成A類對象,系統不知道是選取A類的轉換構造函數進行構造處理,還是選擇B類中的轉換運算符號重載函數處理,系統拒絕從他們兩個中選一個,所以編譯錯誤。   我們修改tp(bt)為tp(A::A(bt));編譯正常,因為我們顯式的明確的告訴系統應該使用A類的轉換構造函數處理,所以,顯式的告訴計算機應該如何處理數據,通常可以解決多義性問題。
美眾議院議長啟動對拜登的彈劾調查
 百态   2023-09-13
上海、濟南、武漢等多地出現不明墜落物
 探索   2023-09-06
印度或要將國名改為「巴拉特」
 百态   2023-09-06
男子為女友送行,買票不登機被捕
 百态   2023-08-20
手機地震預警功能怎麽開?
 干货   2023-08-06
女子4年賣2套房花700多萬做美容:不但沒變美臉,面部還出現變形
 百态   2023-08-04
住戶一樓被水淹 還衝來8頭豬
 百态   2023-07-31
女子體內爬出大量瓜子狀活蟲
 百态   2023-07-25
地球連續35年收到神秘規律性信號,網友:不要回答!
 探索   2023-07-21
全球鎵價格本周大漲27%
 探索   2023-07-09
錢都流向了那些不缺錢的人,苦都留給了能吃苦的人
 探索   2023-07-02
倩女手遊刀客魅者強控制(強混亂強眩暈強睡眠)和對應控制抗性的關系
 百态   2020-08-20
美國5月9日最新疫情:美國確診人數突破131萬
 百态   2020-05-09
荷蘭政府宣布將集體辭職
 干货   2020-04-30
倩女幽魂手遊師徒任務情義春秋猜成語答案逍遙觀:鵬程萬裏
 干货   2019-11-12
倩女幽魂手遊師徒任務情義春秋猜成語答案神機營:射石飲羽
 干货   2019-11-12
倩女幽魂手遊師徒任務情義春秋猜成語答案昆侖山:拔刀相助
 干货   2019-11-12
倩女幽魂手遊師徒任務情義春秋猜成語答案天工閣:鬼斧神工
 干货   2019-11-12
倩女幽魂手遊師徒任務情義春秋猜成語答案絲路古道:單槍匹馬
 干货   2019-11-12
倩女幽魂手遊師徒任務情義春秋猜成語答案鎮郊荒野:與虎謀皮
 干货   2019-11-12
倩女幽魂手遊師徒任務情義春秋猜成語答案鎮郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手遊師徒任務情義春秋猜成語答案鎮郊荒野:指鹿為馬
 干货   2019-11-12
倩女幽魂手遊師徒任務情義春秋猜成語答案金陵:小鳥依人
 干货   2019-11-12
倩女幽魂手遊師徒任務情義春秋猜成語答案金陵:千金買鄰
 干货   2019-11-12
倩女幽魂手遊師徒任務情義春秋猜成語答案金陵:倒履相迎
 干货   2019-11-12
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有