C++继承(2) - 虚函数与运行时多态

目录

1.运行时多态

2.虚函数作用

3.虚函数原理


1.运行时多态

下面就是运行时多态的简单例子。

#include<iostream>
using namespace std;
class Base
{
public:
    virtual void show() { cout<<" In Base \n"; }
};
  
class Derived: public Base
{
public:
    void show() { cout<<"In Derived \n"; }
};

int main(void)
{
    Base *bp = new Derived;
    bp->show();  // 运行时多态
    return 0;
}

运行结果:
In Derived

上面程序中最主要的一点是,使用了一个基类指针来调用子类的成员函数。虚函数的调用不依赖调用它的指针或引用本身的类型,而取决于引用或指针所指向的对象类型。

2.虚函数作用

虚函数允许用户创建一系列基类指针并且调用子类的方法,不需要知道子类对象的类型。
例如,某公司的员工管理系统,存在一个基类Employee,包括多个虚函数如raiseSalary(),transfer(),promote()等。不同类型的员工如Manager,Engineer等,可能有它们自己独立的虚函数实现。
这样使用此系统时,只需传递一系列employees,并调用合适的方法。例如,当需要加工资时(raiseSalary),只需要遍历这个employees列表即可。每个员工可能有它自己的实现逻辑,但用户不需要关注。

class Employee
{
public:
    virtual void raiseSalary() {}
    virtual void promote() {}
};

class Manager: public Employee 
{
    virtual void raiseSalary() {}
    virtual void promote() {}
};

void globalRaiseSalary(Employee* emp[], int n)
{
    for(int i = 0; i < n; i++) {
        emp[i]->raiseSalary(); // 多态调用: 调用raiseSalary()。
                               // 根据的是实际对象, 而不是指针类型
    }
}

像globalRaiseSalary()函数,基于一系列employee对象基础上,可以实现很多的操作,无需知道实际对象的类型。

虚函数机制是非常有用的,像Java语言就默认所有方法都为虚函数。

3.虚函数原理

编译器如何处理这种运行时才确定的行为呢?主要依靠两个方面来支持这种行为:

vtable: 记录函数指针的一张表。

每个类维护一个虚表。

vptr: 指向虚表vtable的指针。

每个对象拥有一个vptr。

 

编译器会在两个地方添加额外代码来维护以及使用vptr。

1)每个构造函数中添加代码
这份代码会在每个对象中创建一个vptr,并且将vptr指向类的vtable。

2)多态函数调用时添加代码
例如,第1节代码中的bp->show()。
只要一个多态的调用出现在某个地方,编译器会在那里插入代码来首先查找vptr,使用基类指针或引用(上面代码中,因为指针所指向的是子类,则访问的是子类的vptr)。
一旦获取到了vptr,子类的vtable就能访问到了。使用vtable,就能访问并调用到子类的show()函数了。

这种机制是否为C++实现运行时多态的标准方法?
C++标准并没有明确要求如何实现运行时多态,但是各编译器基本上都基于这种机制,加入小的修改。

更多参考:
http://en.wikipedia.org/wiki/Virtual_method_table
http://en.wikipedia.org/wiki/Virtual_function

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值