青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

woaidongmao

文章均收錄自他人博客,但不喜標題前加-[轉貼],因其丑陋,見諒!~
隨筆 - 1469, 文章 - 0, 評論 - 661, 引用 - 0
數據加載中……

C++ 轉換運算符重載

大家知道對于內置類型的數據我們可以通過強制轉換符的使用來轉換數據,例如(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);完全相同。
  運行結果如下圖示(注意觀察內存地址,觀察構造與析構過程,執行過程圖中有解釋):

clip_image001

  當一個類含有轉換運算符重載函數的時候,有時候會破壞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對象進行構造,可見中間的創建臨時對象和載入轉換運算符函數的過程完全是多余,讀者對此例要認真解讀,充分理解。
  運行結果如下圖所示(運行過程的解釋見圖):

clip_image002

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

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

//例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"); 
}

  運行過程見下圖。

clip_image003

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

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

  優點:在不提供帶有類對象參數的運算符重載函數的情況下,轉換運算符重載函數可以將類對象轉換成需要的類型,然后進行運算,最后在構造成類對象,這一點和類的運算符重載函數有相同的功效。(例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

   
public
        A(B &);
//轉換構造函數,他的作用是用B類對象構造A類對象 
       
void Edita(int temp) 
        { 
            A::a=temp; 
        } 
   
public
       
int a; 
}; 
class

   
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類的轉換構造函數處理,所以,顯式的告訴計算機應該如何處理數據,通常可以解決多義性問題

 

posted on 2008-09-22 11:49 肥仔 閱讀(1157) 評論(1)  編輯 收藏 引用 所屬分類: C++ 基礎

評論

# re: C++ 轉換運算符重載  回復  更多評論   

給力!!!
2011-03-26 08:20 | tianwen
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            午夜精品久久久久久99热| 麻豆精品一区二区综合av| 国产拍揄自揄精品视频麻豆| 午夜国产一区| 欧美一区二区日韩| 黄色一区二区在线| 欧美成人久久| 欧美日本乱大交xxxxx| 亚洲调教视频在线观看| 亚洲永久免费视频| 国产一区激情| 欧美激情按摩在线| 欧美日在线观看| 久久爱www.| 久久亚洲综合色| 一本久久综合亚洲鲁鲁| 亚洲一二区在线| 精品成人久久| 亚洲精品一区久久久久久| 国产精品美女久久久| 久久亚洲私人国产精品va| 欧美成人蜜桃| 午夜精品视频在线观看| 久久久国产精品亚洲一区 | 国产欧美韩国高清| 免费成人在线观看视频| 欧美精品一区三区| 久久成人人人人精品欧| 美女福利精品视频| 亚洲欧美日韩国产成人精品影院 | 小黄鸭视频精品导航| 久久久久国产成人精品亚洲午夜| 亚洲精品日产精品乱码不卡| 亚洲午夜激情| 18成人免费观看视频| 一本色道久久综合亚洲91| 国模吧视频一区| 亚洲精品色婷婷福利天堂| 国产欧美日韩在线| 亚洲国产一区二区三区青草影视 | 亚洲第一搞黄网站| 99re这里只有精品6| 国语自产精品视频在线看8查询8 | 国产视频在线一区二区 | 亚洲黑丝在线| 亚洲一区二区3| 亚洲人成在线影院| 午夜在线视频观看日韩17c| 日韩视频在线一区| 欧美在线1区| 亚洲视频导航| 毛片基地黄久久久久久天堂| 欧美一区二区三区久久精品| 欧美激情按摩在线| 久久婷婷蜜乳一本欲蜜臀| 欧美调教vk| 亚洲高清在线视频| 国产永久精品大片wwwapp| 日韩亚洲国产欧美| 亚洲国产精品va在看黑人| 亚洲综合电影| 在线视频你懂得一区二区三区| 久久人91精品久久久久久不卡| 亚洲欧美日韩在线高清直播| 欧美夫妇交换俱乐部在线观看| 久久九九99| 国产精品久久久久久影院8一贰佰| 亚洲国产va精品久久久不卡综合| 国产一二精品视频| 中文在线一区| 一本大道久久精品懂色aⅴ| 久久久一二三| 久久久福利视频| 国产精品免费小视频| 日韩视频一区二区在线观看 | 亚洲淫性视频| 一区二区高清视频| 欧美不卡激情三级在线观看| 久久综合久久久| 国产亚洲免费的视频看| 亚洲视频一区二区| 中日韩视频在线观看| 欧美成人高清视频| 欧美va亚洲va香蕉在线| 激情91久久| 久久精品水蜜桃av综合天堂| 欧美专区第一页| 国产精品日韩电影| 亚洲无人区一区| 亚洲一区二区三区777| 欧美日韩精品在线观看| 亚洲国产精选| 亚洲精品久久久蜜桃| 狼狼综合久久久久综合网 | 久久免费少妇高潮久久精品99| 久久国产视频网| 国产麻豆91精品| 亚洲自拍偷拍网址| 午夜在线a亚洲v天堂网2018| 国产精品护士白丝一区av| av成人免费在线观看| 一区二区三区久久| 欧美日韩一二区| 日韩一级二级三级| 亚洲一区二区3| 国产精品久久久久999| 一区二区电影免费观看| 亚洲一区二区三区久久| 国产精品二区影院| 亚洲一区二区三区精品视频 | 国产精品免费一区二区三区在线观看 | 亚洲毛片在线看| 一区二区日韩精品| 欧美日韩亚洲高清| 一区二区三区偷拍| 午夜精品在线| 国产一区二区电影在线观看| 欧美亚洲色图校园春色| 久久久久久久综合色一本| 国内精品伊人久久久久av一坑| 久久精品国产综合精品| 蜜臀av国产精品久久久久| 亚洲国内欧美| 欧美日韩精品三区| 亚洲天堂黄色| 久久国产欧美日韩精品| 狠狠色综合色区| 免费在线观看成人av| 亚洲精品久久嫩草网站秘色| 亚洲视频视频在线| 国产老肥熟一区二区三区| 欧美在线国产| 欧美国产在线观看| 一区二区欧美视频| 国产精品爽爽ⅴa在线观看| 欧美一二区视频| 美女主播一区| 亚洲精选91| 国产精品美女久久福利网站| 久久aⅴ国产欧美74aaa| 欧美激情一区二区三区四区| 99re这里只有精品6| 国产精品久久久久9999吃药| 欧美在线免费观看视频| 欧美黑人国产人伦爽爽爽| 一区二区高清在线观看| 国产手机视频一区二区| 蜜桃av噜噜一区| 一区二区三区波多野结衣在线观看| 欧美一级视频一区二区| 在线观看欧美成人| 欧美日韩hd| 午夜精品福利一区二区三区av | 亚洲午夜久久久久久久久电影院| 久久激情网站| 最新成人av网站| 国产麻豆精品theporn| 玖玖玖免费嫩草在线影院一区| 日韩一级网站| 久久一二三国产| 一区二区三区四区五区精品视频| 国产欧美日韩精品丝袜高跟鞋 | 欧美专区在线播放| 亚洲国产精品久久久久婷婷老年| 午夜精品一区二区三区在线 | 亚洲小说区图片区| 奶水喷射视频一区| 亚洲先锋成人| 在线免费观看视频一区| 国产精品jvid在线观看蜜臀 | 噜噜噜在线观看免费视频日韩| 9人人澡人人爽人人精品| 国产午夜精品理论片a级探花| 欧美大片在线影院| 午夜精品久久久久久久久| 亚洲国产欧美一区二区三区同亚洲| 西西人体一区二区| 亚洲精品一区中文| 国产亚洲欧美激情| 欧美日韩午夜视频在线观看| 久久久久成人精品| 亚洲专区在线视频| 最近中文字幕mv在线一区二区三区四区| 午夜视频在线观看一区二区| 亚洲日本黄色| 国内外成人在线| 国产精品国产自产拍高清av王其| 美女精品在线| 欧美一区二区三区久久精品 | 亚洲视频在线观看一区| 在线不卡视频| 国产精品一区二区三区免费观看| 欧美国产日韩二区| 久久久国产成人精品| 亚洲综合欧美| 99re66热这里只有精品4| 亚洲成人在线免费| 蜜臀av性久久久久蜜臀aⅴ| 性久久久久久久久久久久| 亚洲最新合集|