二、详述条件 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 内。

  好,下面看我一个程序:

 #include <iostream>
#include <string>
using namespace std; class A0
{
public: virtual void fun(){ cout << "I belong to A0!" << endl;}
}; class A: public A0
{
public:
virtual void fun(){ cout << "I belong to A!" << endl;}
}; int main()
{
19 A a;20
20     A b = a;
     return ;
}

  大家看看代码中的第 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函数中的内容变一下:、

 #include <iostream>
#include <string>
using namespace std; class A0
{
public:

9 virtual void fun(){ cout << "I belong to A0!" << endl;}
}; class A: public A0
{
public:
virtual void fun(){ cout << "I belong to A!" << endl;}
}; int main()
{
A a;
A0 a0 = a;
A0 *pa0;
pa0 = &a;
pa0->fun();
pa0 = &a0;
pa0->fun();
return ;
}

  运行结果大家自己测试,如果此时,还是仅仅通过简单的 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就行了。剩下的,我会在后来的博文中阐述清楚!

构造函数语义学之Copy Constructor构建操作(2)的更多相关文章

  1. 构造函数语义学之Copy Constructor构建操作(1)

    一.Copy Constructor的构建操作 就像 default constructor 一样,如果class没有申明一个 copy constructor,就会隐含的声明或隐含的定义一个.生成的 ...

  2. 构造函数语义学之Default Constructor构建操作

    一.Default Constructor的构建操作 首先大家要走出两个误区: 1).任何class如果没有定义default constructor,就会被合成一个来. 2).便以其合成出来的def ...

  3. 深度探索C++对象模型之第二章:构造函数语意学之Copy constructor的构造操作

    C++ Standard将copy constructor分为trivial 和nontrivial两种:只有nontrivial的实例才会被合成于程序之中.决定一个copy constructor是 ...

  4. 构造函数语义学——Copy Constructor 的构造操作

    前言 在三种情况下,会以一个 object 的内容作为另一个 class object 的初值: object明确初始化 class X{...}; X x; X xx = x; object 被当作 ...

  5. 构造函数语义学——Copy Constructor 篇

    构造函数语义学--Copy Constructor 篇 本文主要介绍<深度探索 C++对象模型>之<构造函数语义学>中的 Copy Constructor 构造函数的调用时机 ...

  6. C++ 类 复制构造函数 The Copy Constructor

    一.复制构造函数的定义 复制构造函数是一种特殊的构造函数,具有一般构造函数的所有特性.复制构造函数创建一个新的对象,作为另一个对象的拷贝.复制构造函数只含有一个形参,而且其形参为本类对象的引用.复制构 ...

  7. 构造函数语义学——Default Constructor篇

    构造函数语义学--Default Constructor 篇 这一章原书主要分析了:编译器关于对象构造过程的干涉,即在对象构造这个过程中,编译器到底在背后做了什么 这一章的重点在于 default c ...

  8. 深度探索C++对象模型之第二章:构造函数语意学之Default constructor的构造操作

    C++新手一般由两个常见的误解: 如果任何class没有定义默认构造函数(default constructor),编译器就会合成一个来. 编译器合成的的default constructor会显示的 ...

  9. Copy Constructor的构造操作

    Copy Constructor的构造操作 有三种情况,会以一个object的内容作为另一个class object的初值: 1.  对一个object做显式的初始化操作 class X{…}; X ...

随机推荐

  1. Java 理论与实践: 非阻塞算法简介--转载

    在不只一个线程访问一个互斥的变量时,所有线程都必须使用同步,否则就可能会发生一些非常糟糕的事情.Java 语言中主要的同步手段就是synchronized 关键字(也称为内在锁),它强制实行互斥,确保 ...

  2. css培训(三)

    优先级 z-index 不设置 或auto   非static z-index :0 : z-index:-1: opacity 与层叠上下 opacity:.9 对其影响  小于1值   不具备堆叠 ...

  3. 如何使用Jquery获取Form表单中被选中的radio值

     $("input[name='opType']:checked").val() -------此方法估计用的比较多,通俗易懂 $("input:radio:checke ...

  4. ASP.NET JSON的序列化和反序列化 之 Newtonsoft.Json

    我们用到的类库为:Newtonsoft.Json,通过VS工具中NuGet程序包可以下载. 一:对象转json-序列化 public class Student { public int ID { g ...

  5. hibernate对象关系映射( 一对一,一对多,多对一,多对多的单向,双向映射 ——)

    对象之间的关系: 关系映射之间的关系只的是对象之间的关系,并不指数据库表的关系(外键关系)这儿解决的问题是当对象之间的关系之一时,数据库表该如何映射,编程上如何对待. 一对一(主键关联,和单向的外键关 ...

  6. Axiom3D学习日记 4.地形,天空,雾

    首先需要引用Axiom.SceneManagers.Octree.dll. 地形: 载入地形配置,从一个文件中. scene.LoadWorldGeometry( "Terrain.xml& ...

  7. TextView使用Spannable设置复合文本

    http://blog.csdn.net/linghu_java/article/details/32053167 Spannable 对文字的编辑减少TextView的拼接并且达到改变一串字符中的部 ...

  8. SQL SERVER字符集的研究(中英文字符集,varchar,nvarchar).

    一. 试验归类测试SQL: drop table a )) insert into a values('a') insert into a values(N'a') insert into a val ...

  9. java 对于url地址的实体符号的处理

    <!-- https://mvnrepository.com/artifact/org.apache.commons/commons-lang3 <dependency> <g ...

  10. 第四篇、Tomcat 集群

    1.  前言 该篇中测试的机器发生了变更,在第一篇中设置的Apache  DocumentRoot "d:/deployment"修改为了DocumentRoot d:/clust ...