一、关于多继承
菱形继承:指的是B继承A,C继承A,D继承B和C。下图是菱形继承的过程,以及各类的简单的内存布局。
菱形继承存在的问题:间接基类的数据会出现多份导致访问出错,并且存在内存浪费;我们可以利用虚继承来解决这一问题。
二、虚继承
虚继承是面向对象编程中的一种技术,是指一个指定的基类,在继承体系结构中,将其成员数据实例共享给也从这个基类型直接或间接派生的其它类。虚拟继承是多重继承中特有的概念。虚拟基类是为解决多重继承而出现的。(来自百科的解释)
在虚继承中出现了以下的概念,虚基类指针和虚基类表;
虚基类指针指向虚基类表。虚基类表中有两项,一是vbptr的偏移,二是vbptr所指向的作用域的偏移;
注意:
虚基类的构造优先级高;
虚基类的处理顺序:与继承顺序有关;
三、虚继承的内存布局
多继承下的内存布局:非虚基类布局优先于虚基类;
(1)B虚继承A,C虚继承A,D虚继承B且继承C,以下给出D的内存布局图:
(2)B虚继承A,C虚继承A,D虚继承B且继承C,且虚继承E,给出D的内存布局:
(3)看以下的代码,给出D的内存布局
class A
{
private:
int ma;
};
class B:virtual public A
{
public:
virtual void show(){}
private:
int mb;
};
class C:virtual public A
{
public:
virtual void show(){}
private:
int mc;
};
class D:public B,public C
{
public:
void show(){}
private:
int md;
};
注意:虚函数指针的偏移是相对于整体作用域的。
四、实现一个不能被继承的类
不能被继承的类一是不能被继承,二是能正常使用;
template<typename T>
class Base
{
Base(){}
friend T;
};
class A:virtual public Base<A>
{
public:
A(int a) :ma(a){}
protected:
int ma;
};
class B :public A
{
public:
B(int b) :mb(b), A(b){}
protected:
int mb;
};
int main()
{
A a(10);
B b(10);
return 0;
}
错误信息:
注意友元的使用以及其特点:
一单向性
二不可继承
三不可传递