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

            精产国品久久一二三产区区别 | 久久亚洲国产最新网站| 久久综合九色综合97_久久久| 国产精品禁18久久久夂久| 91性高湖久久久久| 久久精品成人免费观看97| 久久人人爽人人爽人人爽| 久久国产精品一国产精品金尊| 日本精品久久久久中文字幕| 久久久久久亚洲精品无码| 久久中文骚妇内射| 久久人搡人人玩人妻精品首页| 亚洲国产精品无码久久久不卡| 99国内精品久久久久久久| 久久久久久久波多野结衣高潮| 99久久www免费人成精品| 色婷婷综合久久久久中文| 国产精品日韩深夜福利久久| 男女久久久国产一区二区三区 | 91麻豆精品国产91久久久久久| 天天影视色香欲综合久久| 久久国产成人精品麻豆| 亚洲乱码中文字幕久久孕妇黑人| 99久久精品国产免看国产一区| 亚洲а∨天堂久久精品| 久久av免费天堂小草播放| 人妻无码αv中文字幕久久琪琪布| 欧美性猛交xxxx免费看久久久| 久久国产精品久久久| 性欧美丰满熟妇XXXX性久久久| 久久久久久久免费视频| 久久久久九九精品影院| 99精品久久久久久久婷婷| 久久精品国产91久久综合麻豆自制| 日韩av无码久久精品免费| 伊人久久精品无码二区麻豆| 久久精品国产99国产精品亚洲| 日日狠狠久久偷偷色综合0| 久久精品成人欧美大片| 欧美日韩精品久久久久| 久久久国产99久久国产一|