• <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>

            Welcome to ErranLi's Blog!

              C++博客 :: 首頁 :: 聯(lián)系 :: 聚合  :: 管理
              106 Posts :: 1 Stories :: 97 Comments :: 0 Trackbacks

            常用鏈接

            留言簿(12)

            搜索

            •  

            積分與排名

            • 積分 - 175914
            • 排名 - 151

            最新評(píng)論

            閱讀排行榜

                
            如果在派生類中要重載派生類的operator = ,那么在基類中一般不要重載operator = ,因?yàn)檫@樣會(huì)帶來很多麻煩.

            定義了兩個(gè)類:
             class CBase
            {  
            public:
               CBase()
               {
                    cout<<"CBase constructing ..."<<endl;
                }
               virtual ~CBase()
               {
                    cout<<"CBase destructing ..."<<endl;
                }
            public:
               CBase & operator=(const CBase & ent)
               {
                     cout<<"CBase operator = ... "<<endl;
                     return *this;
                }
            };

            class CDerive: public CBase
            {
            public:
                 CDerive()
                 {
                       cout<<"CDerive constructing 
                 }
                 ~CDerive()
                 {
                        cout<<"CDerive destructing ..."<<endl;
                  } 
            public:
                 CDerive & operator=(const CDerive & ent)
                {
                       cout<<"CDerive operator = ... "<<endl;
                       return *this;
                 }
            }

            定義如下操作:

            CBase * b1 = new CDerive();
            CBase * b2 = new CDerive();

            (*b1) = (*b2);

            可以看到其輸出為:
            CBase constructing ...
            CDerive constructing ...
            CBase constructing ...
            CDerive constructing ...
            CBase operator = ...
            CDerive destructing ...
            CBase destructing ...
            CDerive destructing ...
            CBase destructing

            而實(shí)際上,操作(*b1) = (*b2)是想把b1,b2實(shí)際所指類型的兩個(gè)對(duì)象之間進(jìn)行賦值,但是它只是調(diào)用了基類的賦值操作,沒有執(zhí)行其本身派生類的賦值操作。

            發(fā)現(xiàn)有兩種方法可以解決這個(gè)問題,如果能夠知道基類指針實(shí)際所指的是哪個(gè)派生類的的對(duì)象,直接類型轉(zhuǎn)換就可以了:

            (*((CDerive*)b1)) = (*((CDerive*)b2));

            這樣可以發(fā)現(xiàn)輸出了:CDerive operator = ... 。但是當(dāng)要定義基類的指針的時(shí)候,往往隱藏了具體的派生類,即往往不知道指針到底實(shí)現(xiàn)的是哪個(gè)派生類,所以這種方法有很多局限性。

            這時(shí)候可以采用第二種方法,即在派生類中重載基類的賦值操作符,即首先把基類的operator=定義為virtual operator=,再在派生類中重載這個(gè),如:

            CBase & operator=(const CBase & ent)
             {
                   const CDerive * p = reinterpret_cast<const CDerive *>(&ent);
                   return operator=((*p));
             }

            好的,當(dāng)再執(zhí)行 (*b1) = (*b2) 的時(shí)候就可以看到輸出:CDerive operator = ... 。


             在此發(fā)現(xiàn)在派生類中重載派生類的operator =的時(shí)候,如果在基類中也重載operator = ,這樣會(huì)帶來很多麻煩,可能會(huì)丟失很多也許不想丟失的數(shù)據(jù)。
             
            同樣,依次還存在問題:
            CBase * b1 = new CDerive();
            CDervie * d1 = new CDerive();

            (*b1) = (*d1);

            CBase * b2 = new CBase ();
            CDervie * d2 = new CDerive();
            (*b2) = (*d2);

            所以我個(gè)人覺得,如果一個(gè)基類有很多個(gè)派生類的,而派生類中又重載了派生類本身的賦值操作符的時(shí)候,則基類中最好不要去重載賦值操作符,不如直接用函數(shù)去賦值。正如所說那樣,運(yùn)算符重載只是為了一種語法上的方便,是另外一種函數(shù)調(diào)用方式,如果不能帶來方便,就沒必要了,還不如直接用函數(shù)去代替。


            .............敬請(qǐng)指點(diǎn)..................
            posted on 2006-05-22 01:09 erran 閱讀(1651) 評(píng)論(2)  編輯 收藏 引用 所屬分類: C & C++

            Feedback

            # re: C++隨筆 關(guān)于virtual operator = 2006-06-02 08:26 LOGOS
            嗯,在有繼承的體系中,重載賦值操作符的確會(huì)發(fā)生很多災(zāi)難性的事情。
            增加知識(shí)了。  回復(fù)  更多評(píng)論
              

            # re: C++隨筆 關(guān)于virtual operator = 2009-11-23 13:58 Noock Tian
            CBase & operator=(const CBase & ent)
            {
            const CDerive * p = reinterpret_cast<const CDerive *>(&ent);
            return operator=((*p));
            }
            這個(gè)實(shí)現(xiàn)是有問題的,返回值應(yīng)該是
            CDerived& operator=(const CDerived& other)
            {
            CBase::operator=(other);
            // Copy new members in CDerived
            return *this;
            }
              回復(fù)  更多評(píng)論
              

            久久se精品一区精品二区国产| 久久青草国产手机看片福利盒子| 亚洲狠狠综合久久| 久久精品国产WWW456C0M| 亚洲人AV永久一区二区三区久久| 日韩精品久久久久久久电影| 欧洲人妻丰满av无码久久不卡| www.久久热.com| 久久伊人亚洲AV无码网站| 性欧美丰满熟妇XXXX性久久久| 久久久久久免费一区二区三区| 亚洲七七久久精品中文国产| 色噜噜狠狠先锋影音久久| 欧美国产成人久久精品| 久久超碰97人人做人人爱| 久久久久无码专区亚洲av| 欧美大香线蕉线伊人久久| 天堂无码久久综合东京热| 999久久久无码国产精品| 国产精品99久久久精品无码| 夜夜亚洲天天久久| 国产精品99久久99久久久| 久久婷婷五月综合色奶水99啪| 久久精品中文字幕久久| 午夜精品久久久久久毛片| 伊人久久大香线蕉综合热线| 久久国产香蕉一区精品| 久久久久久久尹人综合网亚洲| 亚洲国产精品无码久久久不卡| 一97日本道伊人久久综合影院| 国产2021久久精品| 亚洲狠狠综合久久| 久久av免费天堂小草播放| 久久久久一本毛久久久| 国产精品丝袜久久久久久不卡| 久久伊人精品青青草原高清| 国产精品久久一区二区三区| 久久久久亚洲av无码专区导航| 久久久久亚洲精品日久生情| 久久天天躁狠狠躁夜夜2020一 | 伊人久久大香线蕉综合Av|