c++ 虚函数实现多态的原理

        前面的文章说过了c++如何实现多态,最主要的就是使用虚函数,另一种方法就是RTTI,这里我们不做说明了。前面说过编译器是这样实现晚绑定的:编译器对每一个含有virtual函数的类创建一个虚函数表vtable,其实vtable就是一个函数指针数组,vtable中存放该类的类型信息和这个类所有的虚函数地址,而且在这个类中,编译器会隐含的设置一个指向这个虚函数表的虚拟指针vptr。每个类vptr的设定和重置是通过类的构造函数、析构函数和拷贝复制运算符自动完成。vtable中的type_info object就是支持RTTI的,一般放在vtable的第一个位置。看下面的一段含有继承关系的代码
class Base {
public:
	Base();
	virtual ~Base() {};
	virtual void test() {
		std::cout << "Base::test" << std::endl;
	}

private:
	int b_i;
};

class Derived : public Base {
public:
	virtual void test() {
		std::cout << "Derived::test" << std::endl;
	}

private:
	int d_i;
};


他们的对象在内存布局大概形式如图所示

若是没有虚函数时就不需要中间的vtable间接查找。其中vtable和type_info是放在内存的静态存储区的。vptr的初始化实在类的构造函数中完成,vptr的改写是在类的析构函数中完成。

       使用 Base *b = new Derived;b->test();这种通过基类的引用或指针调用虚函数时,编译器先根据指针或引用的的静态类型来判断所调的虚函数是不是属于该类或者它的某个public基类,进行类型检查,然后改写虚函数调用语句,在本例中改写成如下形式

	(*(b->vptr[2]))(b);			//b->test();

记住,vtable是函数指针数组,通过偏移量调用相应的函数。那是怎么调用到这个重写后的test虚函数呢?主要就在vtable的构造和b指针当前所指向的对象。在运行时刻,b指向的是Derived对象,而该对象的vptr指向的是Derived::vtable,这个vtable中存放的是改写后的虚函数地址,而不是Base的虚函数地址。这样就可以正确调用了。

        由于调用的时候是根据vtable的偏移量决定,所以函数指针数组中的 虚函数指针排列顺序非常重要:

如果一个虚函数在类中是第一次出现,他在vtable中的地址是插入到最后。若果派生类改写了基类的虚函数,这个函数地址在派生类的vtable的位置要和基类中vtable一样(否则通过偏移量找到的不一定是重写的函数地址)。若派生类没有重写基类的虚函数只是单纯继承下来,虚函数在vtable中的位置和基类vtable位置一样。

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
下面是一个使用C++虚函数实现多态的代码示例: ```cpp #include <iostream> using namespace std; class A { public: virtual void foo() { cout << "A::foo()" << endl; } virtual ~A() { cout << "~A()" << endl; } }; class B : public A { public: void foo() { cout << "B::foo()" << endl; } ~B() { cout << "~B()" << endl; } }; int main() { A* p = new B(); p->foo(); delete p; return 0; } ``` 在这个示例中,类A和类B都有一个名为foo()的虚函数。当我们通过基类指针p调用foo()函数时,实际上会根据指针所指向的对象的类型来确定调用哪个类的foo()函数。这就是多态实现原理。 输出结果为: ``` B::foo() ~B() ~A() ``` 可以看到,通过虚函数实现多态,调用的是派生类B的foo()函数,而不是基类A的foo()函数。同时,析构函数也是虚函数,确保在删除指针p时,会先调用派生类B的析构函数,再调用基类A的析构函数。这是因为在C++中,如果基类的析构函数不是虚函数,那么通过基类指针删除派生类对象时,只会调用基类的析构函数,而不会调用派生类的析构函数。因此,为了确保正确的析构顺序,析构函数应该声明为虚函数。 请注意,虚函数只能在类的成员函数中声明,构造函数不能声明为虚函数。这是因为在创建对象时,需要先调用构造函数来初始化对象,而虚函数的机制是在运行时根据对象的类型来确定调用哪个函数,而构造函数在对象创建时就已经确定了。因此,构造函数不能声明为虚函数。 #### 引用[.reference_title] - *1* [虚函数实现多态原理](https://blog.csdn.net/qq_24309981/article/details/89102183)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down1,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [C++虚函数多态实现](https://blog.csdn.net/qq_27576655/article/details/124535530)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值