当前位置: 首页 > news >正文

从静态多态、动态多态到虚函数表、虚函数指针

        多态(Polymorphism)是面向对象编程中的一个重要概念,它允许不同类的对象对同一消息做出不同的响应。多态性使得可以使用统一的接口来操作不同类的对象,从而提高了代码的灵活性和可扩展性。

一、多态的表现形式

1. 静态多态(编译时多态)

        静态多态主要通过函数重载、运算符重载以及模板来实现。通过不同的参数列表、泛型类来选择合适的函数。

重载

#include <iostream>void print(int a) {std::cout << "Integer: " << a << std::endl;
}void print(double a) {std::cout << "Double: " << a << std::endl;
}int main() {print(10);    // 调用 void print(int)print(10.5);  // 调用 void print(double)return 0;
}

模板

#include <iostream>template <typename T>
void print(T a) {std::cout << "Value: " << a << std::endl;
}int main() {print(10);    // 生成 void print<int>(int)print(10.5);  // 生成 void print<double>(double)return 0;
}
2. 动态多态(运行时多态)

        动态多态主要由虚函数和继承来实现,根据对象的实际类型来调用相应的函数。

#include <iostream>class Base {
public:virtual void print() const {    // 如果这里将virtual注释掉,下面两个都会输出"Base"std::cout << "Base" << std::endl;}virtual ~Base() {} // 虚析构函数
};class Derived : public Base {
public:void print() const override {std::cout << "Derived" << std::endl;}
};int main() {Base* basePtr = new Base();Base* derivedPtr = new Derived();basePtr->print();     // 输出 "Base"derivedPtr->print();  // 输出 "Derived"delete basePtr;delete derivedPtr;return 0;
}

        上面的代码中又提到了把virtual注释掉的情况,这涉及到了 “静态类型” “动态类型” 。在这一部分结束之后会讲到。

二、虚函数表、虚函数指针

        虚函数通过运行时的动态绑定,来实现在子类中重写基类的函数。虚函数的原理可以通过虚函数表、虚函数指针来解释。

1. VTable和vptr

        ·每个包含虚函数的类,都有一个虚函数表VTable,是一个指向函数指针的数组。

        ·每个对象创建之后,会有一个虚函数指针vptr,指向类的虚函数表VTable。

        ·当调用虚函数时,会通过vptr查找VTable,然后调用对应的函数。

2. 构造函数不能是虚函数

        在对象创建时,编译器会给对象的vptr赋值,然后再调用构造函数,如果构造函数是虚函数,此时就陷入了死循环。

3. 析构函数可以是虚函数

        通过将基类的析构函数声明为虚函数,可以确保在通过基类指针删除子类对象时,调用到子类的析构函数,合适地释放资源。

#include <iostream>class Base {
public:virtual ~Base() {std::cout << "Base destructor" << std::endl;}
};class Derived : public Base {
public:~Derived() {std::cout << "Derived destructor" << std::endl;}
};int main() {Base* ptr = new Derived();delete ptr; // 先调用 Derived 的析构函数,然后再调用 Base 的析构函数return 0;
}

三、静态类型、动态类型

        静态类型:是指对象在声明时的类型,在编译期已既定。

        动态类型:一个指针、引用目前指向的对象的类型,在运行时确定的。

再来看我们刚才的代码。

        在函数调用时,虚函数会根据动态类型来调用,而普通函数就通过静态类型

#include <iostream>class Base {
public:/*virtual*/ void print() const {    // 将virtual注释掉,下面两个都会输出"Base"std::cout << "Base" << std::endl;}
};class Derived : public Base {
public:    void print() const /*override*/ {    // override和上面的virtual对应std::cout << "Derived" << std::endl;}
};int main() {Base* basePtr = new Base();        // 静态类型Base,动态类型BaseBase* derivedPtr = new Derived();  // 静态类型Base,动态类型DerivedDerived* thirdPtr = new Derived(); // 静态类型Derived,动态类型DerivedbasePtr->print();     // 输出 "Base"derivedPtr->print();  // 输出 "Base"thirdPtr->print();    // 输出 "Derived"delete basePtr;delete derivedPtr;delete thirdPtr;return 0;
}

四、static_cast和dynamic_cast的安全与否

1. static_cast

static_cast是一种显式类型转换,主要用于已知的类型转换。

  • 向上转型(从派生类指针或引用转换为基类指针或引用)是安全的,因为派生类对象可以被视为基类对象的一个特例。
  • 基本类型转换(如 int 到 double)也是安全的。
  • 不进行运行时检查,因此在某些情况下可能会导致未定义行为,特别是当进行向下转型时。
2. dynamic_cast

dynamic_cast是一种运行时类型检查的类型转换,主要用于多态类型之间的转换。

  • 向上转型(从派生类指针或引用转换为基类指针或引用)是安全的。
  • 向下转型(从基类指针或引用转换为派生类指针或引用)是安全的,因为它会在运行时检查类型的有效性。
  • 运行时检查类型安全,如果转换不成功,会返回 nullptr(对于指针)或抛出 std::bad_cast 异常(对于引用)。

 

相关文章:

  • 基于JAVA+SpringBoot+Vue的疫苗发布和接种预约系统
  • 认知世界的经济学读书笔记
  • slam典型应用手搓
  • 暴雨讲堂:算力高速互联催化超节点开启AI新篇章
  • Python知识点:如何使用Python进行无人机数据处理
  • Gstreamer中,使用mp4或者flv作为视频源去推流RTP等视频流时,需要先解码在编码才能正常
  • uniapp view设置当前view之外的点击事件
  • 类与对象—python
  • Anaconda教程
  • Kubernetes服务发布基础
  • LeetCode 149. 直线上最多的点数
  • LaTeX 编辑器-TeXstudio
  • 【计算机网络最全知识点问答】第二章 物理层
  • gitlab-runner集成CI/CD完整项目部署
  • 凤凰模拟器V6中无人机如何设置“有头模式”
  • Angular4 模板式表单用法以及验证
  • Angular数据绑定机制
  • hadoop入门学习教程--DKHadoop完整安装步骤
  • JDK9: 集成 Jshell 和 Maven 项目.
  • Python_OOP
  • python学习笔记-类对象的信息
  • TypeScript迭代器
  • vue总结
  • 从PHP迁移至Golang - 基础篇
  • 从零开始学习部署
  • 通过几道题目学习二叉搜索树
  • 一、python与pycharm的安装
  • Java性能优化之JVM GC(垃圾回收机制)
  • 宾利慕尚创始人典藏版国内首秀,2025年前实现全系车型电动化 | 2019上海车展 ...
  • ​水经微图Web1.5.0版即将上线
  • $jQuery 重写Alert样式方法
  • (aiohttp-asyncio-FFmpeg-Docker-SRS)实现异步摄像头转码服务器
  • (编程语言界的丐帮 C#).NET MD5 HASH 哈希 加密 与JAVA 互通
  • (汇总)os模块以及shutil模块对文件的操作
  • (接口封装)
  • (切换多语言)vantUI+vue-i18n进行国际化配置及新增没有的语言包
  • (十六)、把镜像推送到私有化 Docker 仓库
  • (心得)获取一个数二进制序列中所有的偶数位和奇数位, 分别输出二进制序列。
  • (学习日记)2024.03.12:UCOSIII第十四节:时基列表
  • (原創) 如何優化ThinkPad X61開機速度? (NB) (ThinkPad) (X61) (OS) (Windows)
  • (转)linux 命令大全
  • ***php进行支付宝开发中return_url和notify_url的区别分析
  • ***通过什么方式***网吧
  • *Django中的Ajax 纯js的书写样式1
  • .mp4格式的视频为何不能通过video标签在chrome浏览器中播放?
  • .NET Core MongoDB数据仓储和工作单元模式封装
  • .Net Core 生成管理员权限的应用程序
  • .NET Core跨平台微服务学习资源
  • .NET Core中如何集成RabbitMQ
  • .NET delegate 委托 、 Event 事件,接口回调
  • .net Stream篇(六)
  • .Net6 Api Swagger配置
  • .NetCore项目nginx发布
  • .net操作Excel出错解决
  • @ 代码随想录算法训练营第8周(C语言)|Day57(动态规划)