给出如下代码段:

  1. #include <iostream>
  2. #include "stdio.h"
  3. using namespace std;
  4. class A
  5. {
  6. public:
  7. A(int arg):m_a(arg)
  8. {
  9. cout << "constructor of A" << endl;
  10. output();
  11. }
  12. virtual void output()
  13. {
  14. cout << "output A" << endl;
  15. }
  16. virtual void display()
  17. {
  18. output();
  19. }
  20. private:
  21. int m_a;
  22. };
  23. class B : public A
  24. {
  25. public:
  26. B(int arg1, int arg2):A(arg1), m_b(arg2)
  27. {
  28. cout << "constructor of B" << endl;
  29. }
  30. virtual void output()
  31. {
  32. cout << "output B" << endl;
  33. }
  34. private:
  35. int m_b;
  36. };
  37. int main(int argc, char* argv[])
  38. {
  39. B b(1, 2);
  40. b.display();
  41. return 0;
  42. }

这段代码会输出什么?

其实这个问题本质是在A的构造函数及display函数中分别调用的是A类的output函数还是B类的output函数。

输出结果是:

constructor of A
output A
constructor of B
output B

说明A的构造函数中调用的是A类的output函数,display函数中调用的是B类的output函数。

为什么会这样呢?这跟指向虚函数表的指针(vptr)初始化时间有关。

《Inside the c++ Object model》中指出vptr初始化的时间为:

After invocation of the base class constructors but before execution of user-provided code

or the expansion of members initialized within the member initialization list.

意思是在所有基类构造函数之后,但又在自身构造函数或初始化列表之前。

vptr初始化是在初始化列表之前还是之后是跟编译器实现有关的,在VC++6.0编译器中是在初始化列表之后,从上面这段代码的部分汇编码就可以看出。

B类的构造函数的汇编码:

00401250   push        ebp
00401251   mov         ebp,esp
00401253   sub         esp,44h
00401256   push        ebx
00401257   push        esi
00401258   push        edi
00401259   push        ecx
0040125A   lea         edi,[ebp-44h]
0040125D   mov         ecx,11h
00401262   mov         eax,0CCCCCCCCh
00401267   rep stos    dword ptr [edi]
00401269   pop         ecx
0040126A   mov         dword ptr [ebp-4],ecx
0040126D   mov         eax,dword ptr [ebp+8]
00401270   push        eax
00401271   mov         ecx,dword ptr [ebp-4]
00401274   call        @ILT+15(A::A) (00401014)                                               //调用基类构造函数
00401279   mov         ecx,dword ptr [ebp-4]
0040127C   mov         edx,dword ptr [ebp+0Ch]
0040127F   mov         dword ptr [ecx+8],edx
00401282   mov         eax,dword ptr [ebp-4]                                                        //这里初始化m_b
00401285   mov         dword ptr [eax],offset B::`vftable' (0042f030)                //初始化vptr
30:           cout << "constructor of B" << endl;
0040128B   push        offset @ILT+50(std::endl) (00401037)
00401290   push        offset string "constructor of B" (0042f01c)
00401295   push        offset std::cout (00433ea0)
0040129A   call        @ILT+180(std::operator<<) (004010b9)
0040129F   add         esp,8
004012A2   mov         ecx,eax
004012A4   call        @ILT+135(std::basic_ostream<char,std::char_traits<char> >::operator<<) (0040108c)

A类构造函数汇编码:

004012E0   push        ebp
004012E1   mov         ebp,esp
004012E3   sub         esp,44h
004012E6   push        ebx
004012E7   push        esi
004012E8   push        edi
004012E9   push        ecx
004012EA   lea         edi,[ebp-44h]
004012ED   mov         ecx,11h
004012F2   mov         eax,0CCCCCCCCh
004012F7   rep stos    dword ptr [edi]
004012F9   pop         ecx
004012FA   mov         dword ptr [ebp-4],ecx
004012FD   mov         eax,dword ptr [ebp-4]
00401300   mov         ecx,dword ptr [ebp+8]
00401303   mov         dword ptr [eax+4],ecx
00401306   mov         edx,dword ptr [ebp-4]                                         //初始化m_a
00401309   mov         dword ptr [edx],offset A::`vftable' (0042f050)  //初始化vptr
10:           cout << "constructor of A" << endl;
0040130F   push        offset @ILT+50(std::endl) (00401037)
00401314   push        offset string "constructor of A" (0042f03c)
00401319   push        offset std::cout (00433ea0)
0040131E   call        @ILT+180(std::operator<<) (004010b9)
00401323   add         esp,8
00401326   mov         ecx,eax
00401328   call        @ILT+135(std::basic_ostream<char,std::char_traits<char> >::operator<<) (0040108c)
11:           output();
0040132D   mov         ecx,dword ptr [ebp-4]
00401330   call        @ILT+195(A::output) (004010c8)

这里还可以看出b对象的vptr是被初始化了两次:

先在基类的构造函数前初始化为指向基类虚函数表(vtble)的指针,然后在自身构造函数前初始化为指向自身类vtble的指针。

而且不管哪种情况,vptr都是在自身构造函数体之前初始化。

所以,在A类的构造函数中调用output函数时,b对象尚未构造完成,vptr指向A类的vtble,当然调用A类的output函数。

当执行b.display时,b对象已经构造完成,vptr指向B类的vtble,当然调用B类的display函数。

Ok,这就解释了前面的输出结果。

接着再引用一个《Inside the c++ object model》中的问题:

Is it safe to invoke a virtual function of the class within its constructor's member initialization list?

我相信聪明的你已经知道答案。

http://blog.csdn.net/passion_wu128/article/details/8580306

C++vptr初始化时间的更多相关文章

  1. 为什么要问Servlet的初始化时间

    Servlet的init方法到底是在什么时候调用的? j2ee specification和java doc中有以下说明 如果load-on-startup设置为>=0, 部署的时候就会调用.  ...

  2. Spark延长SparkContext初始化时间

    有些应用中可能希望先在driver上运行一段java单机程序,然后再初始化SparkContext用集群模式操作java程序返回值.从而避免过早建立SparkContext对象分配集群资源,使资源长时 ...

  3. Unity-Animator深入系列---StateMachineBehaviour初始化时间测试

    回到 Animator深入系列总目录 结果和想的有点出入 测试结果: 1.SMB初始化会被调用多次,次数不可控,当Animator组件重复开关则重复初始化. 2.SMB支持构造函数 MyClass p ...

  4. day03_12/13/2016_bean的管理之作用域与初始化时间

    在Spring中,Bean有几种作用域: 1.singleton作用域 当一个bean的作用域设置为singleton,那么Spring IOC容器中只会存在一个共享的bean实例,并且所有对bean ...

  5. 深入探索c++对象模型

    第一章关于对象 c++在布局和存取时间的额外负担主要有virtual引起 virtual function:运行期动态绑定 virtual base class :base class多次出现在派生类 ...

  6. 【Android自学日记】使用DatePicker以及TimePicker显示当前日期和时间

    DatePicker 1.获取一个日历对象: Calendar cal=Calendar.getInstance(); 2.获取当前日期及时间: int year=cal.get(Calendar.Y ...

  7. iOS时间问题

    在iOS开发中,经常会遇到各种各样的时间问题,8小时时差,时间戳,求时间间隔,农历等等.解决办法网上比比皆是,但大多零零散散,很多资料并没有说明其中问题.这里集中总结一下,以便于以后查阅和供大家参考. ...

  8. C#----对时间结构DateTime的使用(时间日期的使用)

    1.其初始化时间是0001--0--0 0:0:0 class Program { DateTime dt; static void Main(string[] args) { Program pp ...

  9. BW标准数据源初始化设置

    在安装了一干补丁和做好了BW与R3的链接之后(此处有BISIS操心,具体事宜不详),我们就可以登录到R3系统看个究竟了. 磨刀不误砍柴工,先检查一下两边系统的补丁: R3端如下, ,貌似我们是19,通 ...

随机推荐

  1. PHP多种序列化/反序列化的方法 (转载)

    1. serialize和unserialize函数 这两个是序列化和反序列化PHP中数据的常用函数. <?php $a = array('a' => 'Apple' ,'b' => ...

  2. Windows -DOS 下Yii创建应用及出错处理

    Win7下,以管理员身份运行cmd.exe 本例是在wamp环境下:切换到d盘,再切换到framework目录.............1--C:\windows\system32>d: 2-- ...

  3. Python: xml转json

    1,引言 GooSeeker早在9年前就开始了Semantic Web领域的产品化,MS谋数台和DS打数机是其中两个产品.对web内容做结构化转换和语义处理的主要路线是 XML -> RDF - ...

  4. Python2.7 转义和正则匹配中文

    今天爬虫(新浪微博 个人信息页面)的时候遇到了转义和正则匹配中文出乱码的问题. 先给出要匹配的部分网页源代码如下: <span class=\"pt_title S_txt2\&quo ...

  5. ctype.h库函数

    头文件ctype.h声明了一组用于分类和转换单个字符的函数.所有的函数都接收一个int型的参数,并返回一个int——返回的int可能代表一个字符,也可能代表的是bool值(0为假,非0为真). 你可能 ...

  6. 关于采用MVC开发默认路由导致首页部分文件访问失效的临时解决方案

    最近开发中涉及了Mvc4的开发,其中的默认路由功能是很不错的东西,但是在实际应用中就出现了不少的问题.比如我们访问某网站http://www.abc.com,虽然路由会帮助自动转向Home/Index ...

  7. STL set 使用小结

    这是微软帮助文档中对集合(set)的解释: “描述了一个控制变长元素序列的对象(注:set中的key和value是Key类型的,而map中的key和value是一个pair结构中的两个分 量)的模板类 ...

  8. GroundworkCSS ♥ Tables

    微信公众平台开发(41)一键关注微信公众平台账号 - 方倍工作室 - 博客园 微信公众平台开发(41)一键关注微信公众平台账号 GroundworkCSS ♥ Tables Example Layou ...

  9. 写一个将当前页面 URL 中的 get 参数解析成一个对象的方法。

    function getQuery () { var args = {}; var query = window.location.search.substring(1); var pairs = q ...

  10. Play Apple(博弈)

    Problem 1604 - Play Apple Time Limit: 1000MS   Memory Limit: 65536KB    Total Submit: 434  Accepted: ...