c++对象模型
类中成员分类
数据成员分为静态和非静态,成员函数有静态非静态以及虚函数
class data members:static和nonstatic
class data functions:static、nonstatic和virtual
C++对象模型
在此模型下,nonstatic 数据成员被置于每一个类对象中,而static数据成员被置于类对象之外。static与nonstatic函数也都放在类对象之外,而对于virtual 函数,则通过虚函数表+虚指针来支持:
例如
class Base
{
public:
Base(int i) :baseI(i){};
int getI(){ return baseI; }
static void countI(){};
virtual void print(void){ cout << "Base::print()";
virtual ~Base(){}
private:
int baseI;
static int baseS;
};
单继承(父类含虚函数)
原则:对普通单继承而言
子类与父类拥有各自的一个虚函数表
若子类并无overwrite父类虚函数,用父类虚函数
若子类重写(overwrite)了父类的虚函数,则子类虚函数将覆盖虚表中对应的父类虚函数
若子声明了自己新的虚函数,则该虚函数地址将扩充到虚函数表最后
一般多继承
这里讲的是不考虑菱形继承的多继承,因为菱形继承需要用到虚继承,放到之后考虑
原则:
若子类新增虚函数,放在声明的第一个父类的虚函数表中
若子类重写了父类的虚函数,所有父类的虚函数表都要改变:如print()
内存布局中,父类按照其声明顺序排列
简单虚继承
原则:虚继承解决了菱形继承中最派生类拥有多个间接父类实例的情况,在C++对象模型中,虚继承而来的子类会生成一个隐藏的虚基类指针(vbptr)
虚继承的子类,如果本身定义了新的虚函数,则编译器为其生成一个新的虚函数指针(vptr)以及一张虚函数表。该vptr位于对象内存最前面(对比非虚继承:直接扩展父类虚函数表)
虚继承的子类也单独保留了父类的vprt与虚函数表
虚继承的子类有虚基类表指针(vbptr)