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

            那誰的技術博客

            感興趣領域:高性能服務器編程,存儲,算法,Linux內核
            隨筆 - 210, 文章 - 0, 評論 - 1183, 引用 - 0
            數據加載中……

            一個關于臨時對象和虛擬析構函數的問題

            在CU逛,別人問到這樣的一個問題,大致如下:

            #include <iostream>
            #include 
            <string>
            using namespace std;
            class base
            {
                    
            public:
                            
            virtual void print(){cout<<"base::print()"<<endl;}
                            
            //virtual ~base(){}//增加虛析構函數,將導致輸出不一致。
            }
            ;
            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;
            }


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



            我的解釋如下:
            實際上,第一個賦值指向的是一個臨時對象,加入與沒有加入虛擬析構函數的區別在于:加入之后,在derived類中會合成一個析構函數以便調用base的虛擬析構函數,如果沒有加入的話那么就不會合成這個析構函數,所有的資源在main函數結束之后才回收.

            因此,對于代碼: base* p1=&derived();
            沒有加入虛擬析構函數的時候因為derived()函數生成的臨時對象沒有被銷毀,因此對它的調用是對derived的調用;當加入虛擬析構函數之后,derived()函數生成的臨時對象在以上的賦值完成之后就會調用析構函數進行析構,這個時候再次對p1調用print函數時,因為臨時對象已經析構,那么這個調用就是對base的調用了.

            我做了一個實驗的代碼,加了一些東西,大家看看~~

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

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

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

                    
            // 第一部分相當于以下的代碼
                    
            // 用temp模擬臨時對象
                    derived temp;
                    derived 
            *= &temp;
                    
            base *p3 = s;
                    
            // 這次調用是對derived的函數調用
                    p3->print();
                    
            // 顯式地對temp進行析構
                    temp.base::~base();
                    
            // 析構完后,同樣的一個內存,卻是用base指針去解釋,
                    
            // 因此調用的是base的函數
                    p3->print();

                    
            return 0;
            }
             

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

            評論

            # re: 一個關于臨時對象和虛擬析構函數的問題  回復  更多評論   

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

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

            # re: 一個關于臨時對象和虛擬析構函數的問題  回復  更多評論   

            base* p1=&derived();
            這個語句執行完畢,臨時對象已經不存在了。

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

            # re: 一個關于臨時對象和虛擬析構函數的問題[未登錄]  回復  更多評論   

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

            # re: 一個關于臨時對象和虛擬析構函數的問題[未登錄]  回復  更多評論   

            剛才吃西瓜時突然想到:base* p1=&derived(); 后,臨時對象已不復存在,然而編譯器依然會將該臨時對象的內存區解析為base,然而vtbl中的內容依然指向derived::Print(),為base加上析構,這樣vtbl先是指向derived::Print(),而后調用~base()時,該臨時對象的類型變成了base,vtbl表項被糾正為指向base::Print()。
            2008-06-15 21:27 | raof01

            # re: 一個關于臨時對象和虛擬析構函數的問題  回復  更多評論   

            未定義行為,
            base* p1指向一個已經被析構了的對象。

            即使如樓上所說,那也是暫時的運行結果,那時因為編譯器內部實現(vtbl)遺留下的殘骸。

            寫這樣的程序,
            [1] 代價高(臨時對象的構造和析構)
            [2] bt

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

            # re: 一個關于臨時對象和虛擬析構函數的問題  回復  更多評論   

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

            # re: 一個關于臨時對象和虛擬析構函數的問題  回復  更多評論   

            很汗顏!!??!
            2010-04-23 12:44 | 小時候可靚了

            # re: 一個關于臨時對象和虛擬析構函數的問題[未登錄]  回復  更多評論   

            懂不懂的啊
            2010-07-29 08:51 | haha
            国内精品久久久久久中文字幕 | 精品久久久久久国产91| 亚洲国产欧洲综合997久久| 国产亚洲综合久久系列| 久久精品国产99久久丝袜| 7777精品久久久大香线蕉| 久久99国产精品一区二区| 久久久久亚洲AV成人网| 久久久久se色偷偷亚洲精品av| 精品久久久无码人妻中文字幕豆芽| 91久久精品无码一区二区毛片| 国产欧美久久久精品影院| 99久久精品费精品国产一区二区| 国产精久久一区二区三区| 久久强奷乱码老熟女网站| 久久国产精品偷99| 国内精品久久久久久99蜜桃| 色综合久久久久综合99| 国产一区二区三精品久久久无广告 | 无码国内精品久久人妻麻豆按摩| 色狠狠久久AV五月综合| 亚洲国产高清精品线久久| 99久久精品免费国产大片| 91精品国产综合久久婷婷| 无码AV波多野结衣久久| 伊人久久精品影院| 日韩久久无码免费毛片软件 | 国产精品99久久免费观看| 一级做a爰片久久毛片毛片| 国产精品久久久久久久午夜片| 国产午夜精品理论片久久影视| 久久99精品国产自在现线小黄鸭| 亚洲精品无码久久久久sm| 亚洲精品无码成人片久久| 狠狠综合久久AV一区二区三区| 奇米影视7777久久精品人人爽| 午夜精品久久久内射近拍高清| 日本久久中文字幕| 伊人久久五月天| 久久综合88熟人妻| 久久久综合九色合综国产|