C++私有继承与EBO深入分析讲解_第1页
C++私有继承与EBO深入分析讲解_第2页
C++私有继承与EBO深入分析讲解_第3页
C++私有继承与EBO深入分析讲解_第4页
C++私有继承与EBO深入分析讲解_第5页
已阅读5页,还剩2页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

第C++私有继承与EBO深入分析讲解目录私有继承本质不是继承空类大小空基类成员压缩总结Hello!大家好呀,近期逗比老师的一个学生问了我这样一个问题:C++里的私有继承到底有什么意义?

不知道你有没有跟他一样的困惑。的确,我们在编写C++项目中,几乎是没有用过私有继承(这里包括protected继承和private继承),都是清一色的public继承。有的老师干脆直接告诉学生,你见到继承就是public,其他那俩是历史原因,当它不存在就好了。

这种说法呢,其实也有一定道理,但也不全对。对的部分在于:C++中,确实只有public继承才表示的OOP理论中的继承,而私有继承其实对应的是OOP理论中的组合关系,所以说见到继承就写public这话其实没毛病。然而不对的部分在于:私有继承是为了解决某些性能问题而存在的,我们知道通常表示组合的做法是成员对象,但在某些极端情况下,成员对象会出现一些性能问题,这时我们不得不用私有继承来代替。

私有继承本质不是继承

在此强调,这个标题中,第一个继承指的是一种C++语法,也就是classA:B{};这种写法。而第二个继承指的是OOP(面向对象编程)的理论,也就是AisaB的抽象关系,类似于狗继承自动物的这种关系。

所以我们说,私有继承本质是表示组合的,而不是继承关系,要验证这个说法,只需要做一个小实验即可。我们知道最能体现继承关系的应该就是多态了,如果父类指针能够指向子类对象,那么即可实现多态效应。

请看下面的例程:

classBase{};

classA:publicBase{};

classB:privateBase{};

classC:protectedBase{};

voidDemo(){

Aa;

Bb;

Cc;

Base*p=//OK

p=//ERR

p=//ERR

}

这里我们给Base类分别编写了A、B、C三个子类,分别是public、private个protected继承。然后用Base*类型的指针去分别指向a、b、c。发现只有public继承的a对象可以用p直接指向,而b和c都会报这样的错:

CannotcastBtoitsprivatebaseclassBase

CannotcastCtoitsprotectedbaseclassBase

也就是说,私有继承是不支持多态的,那么也就印证了,他并不是OOP理论中的继承关系,但是,由于私有继承会继承成员变量,也就是可以通过b和c去使用a的成员,那么其实这是一种组合关系。或者,大家可以理解为,把b.a.member改写成了b.A::member而已。

那么私有继承既然是用来表示组合关系的,那我们为什么不直接用成员对象呢?为什么要使用私有继承?这是因为用成员对象在某种情况下是有缺陷的。

空类大小

在解释私有继承的意义之前,我们先来看一个问题,请看下面例程

classT{};

//sizeof(T)=

T是一个空类,里面什么都没有,那么这时T的大小是多少?有的同学可能不假思索就会回答0。照理说,空类的大小就是应该是0,但如果真的设置为0的话,会有很严重的副作用,请看例程:

classT{};

voidDemo(){

Tarr[10];

sizeof(arr);//0

T*p=arr+5;

//此时p==arr

p++;//++其实无效

发现了吗?假如T的大小是0,那么T指针的偏移量就永远是0,T类型的数组大小也将是0,而如果它成为了一个成员的话,问题会更严重:

structTest{

Tt;

inta;

//t和a首地址相同

由于T是0大小,那么此时Test结构体中,t和a就会在同一首地址。

所以,为了避免这种0长的问题,编译器会针对于空类自动补一个字节的大小,也就是说其实sizeof(T)是1,而不是0。

这里需要注意的是,不仅是绝对的空类会有这样的问题,只要是不含有非静态成员变量的类都有同样的问题,例如下面例程中的几个类都可以认为是空类:

classA{};

classB{

staticintm1;

staticintf();

classC{

public:

C();

~C();

voidf1();

doublef2(intarg)const;

有了自动补1字节,T的长度变成了1,那么T*的偏移量也会变成1,就不会出现0长的问题。但是,这么做就会引入另一个问题,请看例程:

classEmpty{};

classTest{

Emptym1;

longm2;

//sizeof(Test)==16

由于Empty是空类,编译器补了1字节,所以此时m1是1字节,而m2是8字节,m1之后要进行字节对齐,因此Test变成了16字节。如果Test中出现了很多空类成员,这种问题就会被继续放大。

这就是用成员对象来表示组合关系时,可能会出现的问题,而私有继承就是为了解决这个问题的。

空基类成员压缩

(EBO,EmptyBaseClassOptimization)

在上一节最后的历程中,为了让m1不再占用空间,但又能让Test中继承Empty类的其他内容(例如函数、类型重定义等),我们考虑将其改为继承来实现,EBO就是说,当父类为空类的时候,子类中不会再去分配父类的空间,也就是说这种情况下编译器不会再去补那1字节了,节省了空间。

但如果使用public继承会怎么样?

classEmpty{};

classTest:publicEmpty{

longm2;

//假如这里有一个函数让传Empty类对象

voidf(constEmptyobj){}

//那么下面的调用将会合法

voidDemo(){

Testt;

f(t);//OK

}

Test由于是Empty的子类,所以会触发多态性,t会当做Empty类型传入f中。这显然问题很大呀!如果用这个例子看不出问题的话,我们换一个例子:

classAlloc{

public:

void*Create();

voidDestroy();

classVector:publicAlloc{

//这个函数用来创建buffer

voidCreateBuffer(constAllocalloc){

void*buffer=alloc.Create();//调用分配器的Create方法创建空间

voidDemo(){

Vectorve;//这是一个容器

CreateBuffer(ve);//语法上是可以通过的,但是显然不合理

}

内存分配器往往就是个空类,因为它只提供一些方法,不提供具体成员。Vector是一个容器,如果这里用public继承,那么容器将成为分配器的一种,然后调用CreateBuffer的时候可以传一个容器进去,这显然很不合理呀!

那么此时,用私有继承就可以完美解决这个问题了

classAlloc{

public:

void*Create();

voidDestroy();

classVector:privateAlloc{

private:

void*buffer;

size_tsize;

//...

//这个函数用来创建buffer

voidCreateBuffer(constAllocalloc){

void*buffer=alloc.Create();//调用分配器的Create方法创建空间

voidDemo(){

Vectorve;//这是一个容器

CreateBuffer(ve);//ERR,会报错,私有继承关系不可触发多态

此时,由于私有继承不可触发多态,那么Vector就并不是Alloc的一种,也就是说,从OOP理论上来说,他们并不是继承关系。而由于有了私有继承,在Vector中可以调用Alloc里的方法以及类型重命名,所以这其实是一种组合关系。

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论