c++面向对象编程之继承

今天看了继承这一模块,有点迷迷糊糊,赶快记下来,梳理一下。

c++的类之间有三种继承关系,public,private,protected。对于虚继承,下面也会说明。

例如:class B继承class A

如果B是public继承,则B从A中继承的成员保持着A中的访问权限

如果B是private继承,则B从A中继承的成员(public,protected,private)全变为private成员

如果B是protected继承,则B从A中继承的成员(public,protected)全变为protected,private不变

类所继承成员的访问权限由基类中的成员访问级别和派生类派生列表中使用的访问标号共同控制。这句话有点绕口,举个例子:

class A {

public:

    void fun();

}

class B : private A {

}

int main {

A a;

B b;

a.fun();  //正确

b.fun(); //错误,fun()在B中是private级别的

}

为了实现动态绑定,一般在基类中会声明一个虚函数(virtual),声明虚函数是想让派生类来重定义基类中的虚函数,当然派生类也可以很叛逆,不重定义。

除了构造函数以外,任意的非static函数都可以是虚函数。

如果不想引用派生类B中重定义的虚函数,可以使用域操作符,强制覆盖

再说说虚继承:

普通继承: class B: public A, 
B从A继承,B会将A做一份copy,作为自身的一部分,然后在加自己独有的一部分。
加入A是一个有虚函数的类,考虑A和B的结构
sizeof(A) = sizeof(A的数据成员) + sizeof(vptr),这个vptr指向A类的 vtable
sizeof(B) = sizof(A的数据成员) + sizeof(B数据成员) + sizeof(vptr),这个vptr指向B类的vtable

虚继承: 虚继承与普通继承不同点在于,子类不实际的copy父类的一部分,而是生成一个引用,指向父类对象
但是在用sizeof统计大小时,依然将父类的成员变量统计在内,此时B的结构如下:
sizeof(B) = sizof(A的数据成员) + sizeof(B数据成员) + sizeof(vptr) + sizeof(pointer to A), vptr指向B类的vtable, 并且多了一个引用指向A对象

虚继承的这个特性在C++的多重继承应用中可以起到很好的作用:
一个没有虚继承的多重继承:
class A
{
 void f() ;
};
class B: public A
{
};
class C:public A
{
}
class D: public B, public C
{
}
void main()
{
 D d;
d.f();
}
这种情况下编译器会报错:D::f is ambiguous,could be the 'f' in base 'A' of base 'B' of class 'D',or the 'f' in base 'A' of base 'C' of class 'D'
很明显看出:d.f()是有歧义的,因为D中有2个A的copy,也就有2个A::f()函数,编译器不知道应该调用哪一个。

这个问题应用虚继承可以得到很好的解决,类定义重写如下:
class A
{
 void f() ;
};
class B: public virtual A
{
};
class C:public virtual A
{
}
class D: public B, public C
{
}
void main()
{
 D d;
d.f();
}
此时D中只指向一个A对象,调用d.f(),就是调用那个唯一的A的f()函数,没有歧义,顺利执行。因为此时B和C都是指向同一个A对象。
此时D的结构如下图:

需要注意的是B和C都必须是从A虚继承,只有一个是虚继承仍然会有歧义

参考:http://kuapig.ycool.com/post.2852631.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值