虚函数
class Empty { };
无用的知识1: 空类的大小是1字节.
class Empty {
public:
Empty() = default;
~Empty() = default;
void print() {}
}
无用的知识2: 该类的大小还是1字节,和有多少普通函数没关系。
class Base {
public:
Base() = default;
virtual ~Base() = default;
virtual print() {};
无用的知识3: 该类的大小 sizeof(Base) = sizeof(void*)。
原因很简单,包含了虚函数,所以会包含一个vptr,即虚指针。该指针指向虚函数表,
所谓虚函数表,是编译器自动为一个带有虚函数的类生成的一块内存空间,其中存储着每一个虚函数的入口地址。由于函数的入口地址可以看成一个指针类型,因此这些虚函数的地址间隔为四个字节。而每一个带有虚函数类的实例,都拥有一个虚函数指针——vptr,在类的对象初始化完毕后,它将指向虚函数表。
vptr指针将位于类对象的首部,即作为第一个成员变量。
依据上述理论,我们可以构造一个简单的测试类。demo如下:
using namespace std;
class Base {
public:
Base() = default;
~Base() = default;
virtual void Base1() { cout << "this is Base1 " << endl; }
virtual void Base2() { cout << "this is Base2 " << endl; }
virtual void Base3() { cout << "this is Base3 " << endl; }
};
int main() {
Base base;
using Func = void(*)();
auto vptr = (uintptr_t*)(&base);
auto func = (Func)*(uintptr_t*)(*vptr);
func();
}