zoukankan      html  css  js  c++  java
  • 构造函数语义学之Copy Constructor构建操作(2)

    二、详述条件 3 和 4

      那么好,我又要问大家了,条件1 和 2比较容易理解。因为member object或 base class 含有copy constructor。那么member object所在的class或者base class的derived class需要合成一个nontrivial copy constructor来调用他的member constructor 或 base class的 copy constructor!

      而条件 3 和条件 4比较难理解,我在此阐述一下:

      (1) 条件3:当声明一个或多个virtual functions时

        大家回忆一下,在满足条件 3 时(class 声明了至少一个 virtual function),编译期间的两个程序扩张操作!

        1).增加一个virtual function table ,内涵每一个有作用的function的地址。

        2).将一个指向 virtual function table 的指针(vptr),安插在每一个class object 内。

      好,下面看我一个程序:

     1 #include <iostream>
     2 #include <string>
     3 using namespace std;
     4 
     5 class A0
     6 {
     7 public: 8     virtual void fun(){ cout << "I belong to A0!" << endl;}
    9 };
    10 
    11 class A: public A0
    12 {
    13 public:
    14     virtual void fun(){ cout << "I belong to A!" << endl;}
    15 };
    16 
    17  int main()
    18  {
    19     A a;20     
    20     A b = a;
    21     return 0;
    22  }

      大家看看代码中的第 20 行是不是将 a 赋值给了b,那么 a 中的指针 vptr 是不是 bitwise copy 给了 b 的指针 vptr 呢?答案是肯定的!但是这回指针的bitwise copy 是安全滴。为什么呢?因为同一类型的类实体中的 vptr 在内存中本身就是共享同一个 virtual function table的,请看下面一个图!

        图1 :这张补充说明了程序中 a 和 b 的关系,

      说到这里大家肯定要说了,这个用 bitwise copy semantic 完成不就可以了?为什么还要合成一个copy constructor来完成呢?对,如果仅仅是对付上面代码 bitwise copy semantic 足矣,但是我把上面代码中的main函数中的内容变一下:、

     1 #include <iostream>
     2 #include <string>
     3 using namespace std;
     4 
     5 class A0
     6 {
     7 public:
     8      
    9
    virtual void fun(){ cout << "I belong to A0!" << endl;} 10 }; 11 12 class A: public A0 13 { 14 public: 15 virtual void fun(){ cout << "I belong to A!" << endl;} 16 }; 17 18 int main() 19 { 20 A a; 21 A0 a0 = a; 22 A0 *pa0; 23 pa0 = &a; 24 pa0->fun(); 25 pa0 = &a0; 26 pa0->fun(); 27 return 0; 28 }

      运行结果大家自己测试,如果此时,还是仅仅通过简单的 bitwise copy semantic 的话。那第 24 行中的pa0->fun()将会调用的是 a 的 fun() 输出,I belong to A! 这显然已经不满足类的多态性了嘛!说到这想必大家已经猜到一二了吧?对,此时编译器对类 A0 的对象不能bitwise copy semantic了,而是合成一个copy constructor进行一些必要的操作:包括对程序员定义的 member 的 bitwise copy (浅拷贝)和 对编译器安插的 vptr 进行重新赋值,即指向子类上例中的 A0 class 的 virtual table!很难理解?请看下图:

          图2 :补充说明 a0 和 a 的关系

      对比一下图 1 和图 2,就知道为什么 bitwise copy semantic不行了,而必须要合成一个copy constructor进行进一步操作了吧?

      (2)条件4---当class派生自一个继承串链,其中有一个或多个virtual base classes时

      如果你看懂了,条件 3 那么条件 4 这个也能很快就懂了!大家还记不记得我在我的另一篇博文(构造函数语义学之Default Constructor构建操作)中讲到 virtual base class时讲到:“编译器会为虚继承的derived class中安插一个指针 (vbptr)---用来指向virtual base class pointer table,这个表用来描述从继承类元素到虚基类元素的偏移量。”,既然如此,这和条件3是不是很像呢?也是安插了一个指针,那是不是也不能要bitwise copy constructor 呢?答案是肯定的!因为,这个 vbptr 指针 以及 virtual base class pointer table在内存中如何布局我们还没讲到,这里大家只要知道编译器安插了一个指针,该指针在复制时不能简单的 bitwise copy constructor就行了。剩下的,我会在后来的博文中阐述清楚!

  • 相关阅读:
    vscode 整理————开篇之力(一)
    重学c#系列——datetime 和 datetimeoffset[二十一]
    重新点亮shell————什么是shell[一]
    重新整理 .net core 实践篇——— 权限中间件源码阅读[四十六]
    为什么构建容器需要Namespace?
    基于Windows Mobile 5.0的掌上天气预报设计
    使用.NE平台调用服务访问非托管 DLL 中的函数
    .NET Framework 3.0 RC1 开发环境构建
    ASP.NET未处理异常的处理
    基于Silverlight的Windows Phone 推箱子程序开发
  • 原文地址:https://www.cnblogs.com/zhuwbox/p/3418865.html
Copyright © 2011-2022 走看看