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

            那誰(shuí)的技術(shù)博客

            感興趣領(lǐng)域:高性能服務(wù)器編程,存儲(chǔ),算法,Linux內(nèi)核
            隨筆 - 210, 文章 - 0, 評(píng)論 - 1183, 引用 - 0
            數(shù)據(jù)加載中……

            一個(gè)關(guān)于臨時(shí)對(duì)象和虛擬析構(gòu)函數(shù)的問(wèn)題

            在CU逛,別人問(wèn)到這樣的一個(gè)問(wèn)題,大致如下:

            #include <iostream>
            #include 
            <string>
            using namespace std;
            class base
            {
                    
            public:
                            
            virtual void print(){cout<<"base::print()"<<endl;}
                            
            //virtual ~base(){}//增加虛析構(gòu)函數(shù),將導(dǎo)致輸出不一致。
            }
            ;
            class derived :public base
            {
                    
            public:
                            
            virtual void print(){cout<<"devrived::print()"<<endl;}
            }
            ;
            int main(int argc, char *argv[])
            {
                    
            base* p1=&derived();
                    p1
            ->print();

                    derived d;
                    
            base* p2=&d;
                    p2
            ->print();

                    
            return 0;
            }


            加入虛擬析構(gòu)函數(shù)后,輸出為:
            base::print()
            devrived::print()
            不加入虛擬析構(gòu)函數(shù),輸出則為:
            devrived::print()
            devrived::print()



            我的解釋如下:
            實(shí)際上,第一個(gè)賦值指向的是一個(gè)臨時(shí)對(duì)象,加入與沒(méi)有加入虛擬析構(gòu)函數(shù)的區(qū)別在于:加入之后,在derived類中會(huì)合成一個(gè)析構(gòu)函數(shù)以便調(diào)用base的虛擬析構(gòu)函數(shù),如果沒(méi)有加入的話那么就不會(huì)合成這個(gè)析構(gòu)函數(shù),所有的資源在main函數(shù)結(jié)束之后才回收.

            因此,對(duì)于代碼: base* p1=&derived();
            沒(méi)有加入虛擬析構(gòu)函數(shù)的時(shí)候因?yàn)閐erived()函數(shù)生成的臨時(shí)對(duì)象沒(méi)有被銷毀,因此對(duì)它的調(diào)用是對(duì)derived的調(diào)用;當(dāng)加入虛擬析構(gòu)函數(shù)之后,derived()函數(shù)生成的臨時(shí)對(duì)象在以上的賦值完成之后就會(huì)調(diào)用析構(gòu)函數(shù)進(jìn)行析構(gòu),這個(gè)時(shí)候再次對(duì)p1調(diào)用print函數(shù)時(shí),因?yàn)榕R時(shí)對(duì)象已經(jīng)析構(gòu),那么這個(gè)調(diào)用就是對(duì)base的調(diào)用了.

            我做了一個(gè)實(shí)驗(yàn)的代碼,加了一些東西,大家看看~~

            #include <iostream>
            #include 
            <string>
            using namespace std;
            class base
            {
                    
            public:
                            
            virtual void print(){cout<<"base::print()"<<endl;}
                            
            virtual ~base(){cout << "~base()\n";}//增加虛析構(gòu)函數(shù),將導(dǎo)致輸出不一致。
            }
            ;
            class derived :public base
            {
                    
            public:
                            
            virtual void print(){cout<<"devrived::print()"<<endl;}
                            
            //virtual ~derived(){cout << "~derived()\n";}//增加虛析構(gòu)函數(shù),將導(dǎo)致輸出不一致。
            }
            ;

            int main(int argc, char *argv[])
            {
                    derived 
            *= &derived();
                    
            base* p1 = t;
                    p1
            ->print();

                    derived d;
                    
            base* p2=&d;
                    p2
            ->print();

                    
            // 第一部分相當(dāng)于以下的代碼
                    
            // 用temp模擬臨時(shí)對(duì)象
                    derived temp;
                    derived 
            *= &temp;
                    
            base *p3 = s;
                    
            // 這次調(diào)用是對(duì)derived的函數(shù)調(diào)用
                    p3->print();
                    
            // 顯式地對(duì)temp進(jìn)行析構(gòu)
                    temp.base::~base();
                    
            // 析構(gòu)完后,同樣的一個(gè)內(nèi)存,卻是用base指針去解釋,
                    
            // 因此調(diào)用的是base的函數(shù)
                    p3->print();

                    
            return 0;
            }
             

            posted on 2006-02-16 23:24 那誰(shuí) 閱讀(22339) 評(píng)論(8)  編輯 收藏 引用 所屬分類: C\C++

            評(píng)論

            # re: 一個(gè)關(guān)于臨時(shí)對(duì)象和虛擬析構(gòu)函數(shù)的問(wèn)題  回復(fù)  更多評(píng)論   

            base* p1=&derived();
            后,p1指向的對(duì)象已被銷毀了,p1->???行為是未定義的

            實(shí)際上并不需要virtual ~base(),只要定義~base()就有上述效果了,編譯器會(huì)為這個(gè)析構(gòu)加上糾正vtbl的功能,相比之下,編譯器贈(zèng)送的析構(gòu)函數(shù)卻不會(huì)糾正vtbl
            2006-02-16 23:57 | control

            # re: 一個(gè)關(guān)于臨時(shí)對(duì)象和虛擬析構(gòu)函數(shù)的問(wèn)題  回復(fù)  更多評(píng)論   

            base* p1=&derived();
            這個(gè)語(yǔ)句執(zhí)行完畢,臨時(shí)對(duì)象已經(jīng)不存在了。

            覺得樓主的解釋不合理,但是我也拿不出好的解釋。
            2007-05-18 20:31 | kingsun555

            # re: 一個(gè)關(guān)于臨時(shí)對(duì)象和虛擬析構(gòu)函數(shù)的問(wèn)題[未登錄](méi)  回復(fù)  更多評(píng)論   

            @control
            同意你的觀點(diǎn)。
            博主曾說(shuō)自己的C/C++水平足以應(yīng)付工作。然而,我不得不說(shuō),有很多理解是錯(cuò)的,表面看起來(lái)可以工作,且例子也能論證你的觀點(diǎn),那只不過(guò)是碰巧了而已。而且總是以C的觀點(diǎn)來(lái)看待C++,這是很危險(xiǎn)的。
            2008-06-15 20:26 | raof01

            # re: 一個(gè)關(guān)于臨時(shí)對(duì)象和虛擬析構(gòu)函數(shù)的問(wèn)題[未登錄](méi)  回復(fù)  更多評(píng)論   

            剛才吃西瓜時(shí)突然想到:base* p1=&derived(); 后,臨時(shí)對(duì)象已不復(fù)存在,然而編譯器依然會(huì)將該臨時(shí)對(duì)象的內(nèi)存區(qū)解析為base,然而vtbl中的內(nèi)容依然指向derived::Print(),為base加上析構(gòu),這樣vtbl先是指向derived::Print(),而后調(diào)用~base()時(shí),該臨時(shí)對(duì)象的類型變成了base,vtbl表項(xiàng)被糾正為指向base::Print()。
            2008-06-15 21:27 | raof01

            # re: 一個(gè)關(guān)于臨時(shí)對(duì)象和虛擬析構(gòu)函數(shù)的問(wèn)題  回復(fù)  更多評(píng)論   

            未定義行為,
            base* p1指向一個(gè)已經(jīng)被析構(gòu)了的對(duì)象。

            即使如樓上所說(shuō),那也是暫時(shí)的運(yùn)行結(jié)果,那時(shí)因?yàn)榫幾g器內(nèi)部實(shí)現(xiàn)(vtbl)遺留下的殘骸。

            寫這樣的程序,
            [1] 代價(jià)高(臨時(shí)對(duì)象的構(gòu)造和析構(gòu))
            [2] bt

            別老那java那一套。
            2008-09-12 13:14 | trim

            # re: 一個(gè)關(guān)于臨時(shí)對(duì)象和虛擬析構(gòu)函數(shù)的問(wèn)題  回復(fù)  更多評(píng)論   

            不論是否加入虛擬析構(gòu)函數(shù),臨時(shí)對(duì)象在base* p1=&derived();之后都已經(jīng)析構(gòu)。
            derived對(duì)象的內(nèi)存分布:base類對(duì)象,然后是derived類的虛函數(shù)表和成員變量等(不同編譯器,虛函數(shù)表的位置可能不同)
            析構(gòu)后derived的虛表指針重置,在調(diào)用p1->print()時(shí),只能調(diào)用base類的print()。
            這種調(diào)用存在很大隱患,如果在調(diào)用p1->print()前,其他操作改變了那快內(nèi)存就可能出現(xiàn)其他結(jié)果。
            2010-01-29 00:52 | wcj

            # re: 一個(gè)關(guān)于臨時(shí)對(duì)象和虛擬析構(gòu)函數(shù)的問(wèn)題  回復(fù)  更多評(píng)論   

            很汗顏!!!!
            2010-04-23 12:44 | 小時(shí)候可靚了

            # re: 一個(gè)關(guān)于臨時(shí)對(duì)象和虛擬析構(gòu)函數(shù)的問(wèn)題[未登錄](méi)  回復(fù)  更多評(píng)論   

            懂不懂的啊
            2010-07-29 08:51 | haha
            久久精品国产日本波多野结衣| 久久婷婷五月综合97色直播| 97久久超碰成人精品网站| 久久婷婷五月综合97色| 99久久精品国产一区二区| 日韩十八禁一区二区久久| 精品久久久无码人妻中文字幕 | 久久er热视频在这里精品| 国产精品熟女福利久久AV| 亚洲精品第一综合99久久| 久久精品人人做人人爽电影| 色偷偷88欧美精品久久久| 亚洲成色WWW久久网站| 一级做a爱片久久毛片| 一本色道久久HEZYO无码| 久久国产视频99电影| 久久狠狠高潮亚洲精品| 无码乱码观看精品久久| 国产精品99久久久久久董美香 | 久久99国产精品尤物| 日韩va亚洲va欧美va久久| 久久综合欧美成人| 久久精品午夜一区二区福利| 伊人久久亚洲综合影院| 国产精品99久久久久久www| 国产精品99久久免费观看| 久久精品国产亚洲av麻豆图片| 久久免费视频一区| 国产AⅤ精品一区二区三区久久| 色综合久久中文字幕无码| 精品无码久久久久国产动漫3d| 久久久久九九精品影院| 精品无码人妻久久久久久| 2021少妇久久久久久久久久| 久久久久亚洲AV无码网站| 久久精品国产AV一区二区三区| 久久久久亚洲精品男人的天堂| 精品国产青草久久久久福利| 久久香蕉综合色一综合色88| 国产精品99久久久久久董美香 | 亚洲国产精品久久电影欧美|