两道关于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,那么虚函数表指针就空了,调用的时候找不到对应函数,运行出错!


还有一题

     以下代码的执行情况是:

[cpp]  view plain copy
  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具体原因还没弄清楚。。。难道是派生类对象在基类构造完成后,重新写了虚表指针?如果你知道,还请留言相告啊

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值