博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
C++继承(2) - 虚函数与运行时多态
阅读量:4072 次
发布时间:2019-05-25

本文共 1564 字,大约阅读时间需要 5 分钟。

目录


1.运行时多态

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

#include
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。

 

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

1)每个构造函数中添加代码

这份代码会在每个对象中创建一个vptr,并且将vptr指向类的vtable。

2)多态函数调用时添加代码

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

这种机制是否为C++实现运行时多态的标准方法?

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

更多参考:

你可能感兴趣的文章
MongoDB 数据文件备份与恢复
查看>>
数据库索引介绍及使用
查看>>
MongoDB数据库插入、更新和删除操作详解
查看>>
MongoDB文档(Document)全局唯一ID的设计思路
查看>>
mongoDB简介
查看>>
Redis持久化存储(AOF与RDB两种模式)
查看>>
memcached工作原理与优化建议
查看>>
Redis与Memcached的区别
查看>>
redis sharding方案
查看>>
程序员最核心的竞争力是什么?
查看>>
Node.js机制及原理理解初步
查看>>
linux CPU个数查看
查看>>
分布式应用开发相关的面试题收集
查看>>
简单理解Socket及TCP/IP、Http、Socket的区别
查看>>
利用HTTP Cache来优化网站
查看>>
利用负载均衡优化和加速HTTP应用
查看>>
消息队列设计精要
查看>>
分布式缓存负载均衡负载均衡的缓存处理:虚拟节点对一致性hash的改进
查看>>
分布式存储系统设计(1)—— 系统架构
查看>>
MySQL数据库的高可用方案总结
查看>>