兩道關於C++對象內存模型和多態機制的tricky題目

class parent{
public: virtual void output();  
};  
void parent::output()  
{  
    printf("parent!");  
}  
  
class son : public parent  
{  
public: virtual void output();  
};  
void son::output()  
{  
    printf("son!");  
}


則以下程序段:
son s;
::memset(&s , 0 , sizeof(s));
parent& p = s;
p.output();

執行結果是()

A、parent!       B、son!       C、son!parent!           D、沒有輸出結果,程序運行出錯


答案是D。應該很多人會選擇B,跟我一樣。。。主要是因爲沒看懂memset(&s , 0 , sizeof(s));這行的意思。

      先說下多態吧。基類指針或引用可以引用派生類對象,注意引用不能引用臨時變量,派生類重寫了基類中的虛函數,在使用基類指針或引用調用時就會產生多態行爲,也就是說實際調用的是派生類中的函數。如果去掉memset(&s , 0 , sizeof(s));這一行的話,答案就是B。

      再說一下c++對象內存模型吧。c++編譯器不允許不同對象有相同的內存地址,因此空類和空結構體大小爲1(具體原因)。c++多態的實現機制,含有虛函數的類的每個對象,內部都有一個虛函數地址表指針,每個地址對應代碼段中相應的虛函數。如果採用memset啥都不管就把整個對象的內存全置爲0,那麼虛函數表指針就空了,調用的時候找不到對應函數,運行出錯!


還有一題

     以下代碼的執行情況是:

  1. class classA  
  2. {  
  3. public:  
  4.     classA()  
  5.     {  
  6.         clear();  
  7.     }  
  8.     virtual ~classA()  
  9.     {  
  10.     }  
  11.     void clear()  
  12.     {  
  13.         memset(this , 0 , sizeof(*this));  
  14.     }  
  15.     virtual void func()  
  16.     {  
  17.         printf("func\n");  
  18.     }  
  19. };  
  20. class classB : public classA  
  21. {  
  22. };  
  23.   
  24. int main(void)  
  25. {  
  26.     classA oa;  
  27.     classB ob;  
  28.     classA * pa0 = &oa;  
  29.     classA * pa1 = &ob;  
  30.     classB * pb = &ob;  
  31.   
  32.     oa.func();         // 1  
  33.     ob.func();         // 2  
  34.     pa0->func();       // 3  
  35.     pa1->func();       // 4  
  36.     pb->func();        // 5  
  37.   
  38.     return 0;  
  39. }  
A、func              func        執行出錯      執行出錯        func  
B、執行出錯         func        執行出錯      執行出錯        func 
C、執行出錯      執行出錯      執行出錯      執行出錯      執行出錯
D、func              func          func            func          func
E、func              func        執行出錯          func          func
F、以上選項都不對

    答案是E,語句1、2執行沒問題,3執行肯定出錯,4和5具體原因還沒弄清楚。。。難道是派生類對象在基類構造完成後,重新寫了虛表指針?如果你知道,還請留言相告啊

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章