八、重学C++—动态多态(运行期)

news2025/4/5 16:29:21

上一章节:

七、重学C++—静态多态(编译期)-CSDN博客https://blog.csdn.net/weixin_36323170/article/details/146999362?spm=1001.2014.3001.5502

本章节代码:

cpp/dynamicPolymorphic.cpp · CuiQingCheng/cppstudy - 码云 - 开源中国https://gitee.com/cuiqingcheng/cppstudy/blob/master/cpp/dynamicPolymorphic.cpp

目录

上一章节:

本章节代码:

一、引言

二、概念:什么是运行时多态?

三、实现机制:虚函数

虚函数表和虚函数表指针的作用

四、纯虚函数,抽象类/接口类

抽象类

接口类

五、多态的优点和应用场景

优点

应用场景

六、总结


一、引言

        在 C++ 的奇妙世界里,多态是一个强大而迷人的特性,它就像代码世界中的变形金刚,让程序能够根据不同的情况展现出多样的行为。而 运行期多态,作为多态的重要组成部分 ,更是为程序带来了无与伦比的灵活性和扩展性。今天,我们就一起来深入探秘 C++ 运行期多态的奥秘。

二、概念:什么是运行时多态?

        运行期多态,也被称为 动态多态 ,是指在 程序运行时才确定具体要调用哪个函数 。它通过 虚函数和继承机制来实现,允许我们使用基类的指针或引用调用派生类的函数,从而实现不同对象的不同行为 。简单来说,就是在运行时 根据实际对象的类型来决定执行哪个函数 ,就像变形金刚在战斗中根据不同的场景变换形态一样。

三、实现机制:虚函数

        虚函数是实现运行期多态的关键。在基类中,我们使用 “ virtual 关键字来声明虚函数, 派生类可以重写这些虚函数以实现自己的特定行为 。当通过基类的指针或引用调用虚函数时,程序会在运行时根据指针或引用实际指向的对象类型来决定调用哪个版本的函数。
class base{
    public:
        virtual void func() // 虚函数
        { }
}

class A:public base{
    public:
        void func() override 
        {
            // 重写实现虚函数        
        }
}
每一个 包含虚函数的类中均有一个虚函数表指针,指向虚函数表;如下图:

虚函数表和虚函数表指针的作用

虚函数表( VTable): 每个 包含虚函数的类都有一个虚函数表,这是一个存储虚函数地址的数组 。表中的每个条目都是一个指向虚函数的指针,这些虚函数按照它们在类中声明的顺序排列。
虚函数表指针( VPTR): 每个包含虚函数的类的对象都有一个虚函数表指针,它指向该类对应的虚函数表 。借助这个指针,对象能够在运行时找到正确的虚函数来调用。
实例:一个动物的基类,均有叫这个实现函数,对于两个继承自动物类的狗类/猫类,分别实现其叫这个函数,代码如下:
/***
 *  C++ 多态
 *      动态多态
 *          虚函数/纯虚函数
 *          抽象类/接口类
 */

 #include <iostream>

// 基类 Animal
class Animal {
public:
    // 虚函数 speak()
    virtual void speak() {
        std::cout << "动物发出声音" << std::endl;
    }
};

// 派生类 Cat
class Cat : public Animal {
public:
    // 重写 speak() 函数
    void speak() override {
        std::cout << "喵~" << std::endl;
    }
};

// 派生类 Dog
class Dog : public Animal {
public:
    // 重写 speak() 函数
    void speak() override {
        std::cout << "汪!" << std::endl;
    }
};

int main() {
    // 创建 Cat 和 Dog 对象
    Cat cat;
    Dog dog;

    // 使用基类指针指向派生类对象
    Animal* animal1 = &cat;
    Animal* animal2 = &dog;

    // 调用虚函数
    animal1->speak(); // 输出:喵~
    animal2->speak(); // 输出:汪!

    animal1->Animal::speak(); // 输出:动物发出声音
    return 0;
}
这里 基类的指针,会根据实际指向的对象类型,从而决定默认调用的是重写的虚函数 ,若还想调用基类原来的虚函数,需要指定作为基类方可调用;

四、纯虚函数,抽象类/接口类

上面定义的基类中,虚函数还是实现了,若 一个虚函数没有实现的代码段体,则是纯虚函数,定义如下:
virtual int func() = 0;

抽象类

        包含纯虚函数的类被称为抽象类 抽象类不能被实例化 ,只能作为基类被派生类继承。 派生类必须重写抽象类中的纯虚函数,否则派生类也会成为抽象类。只有实现了纯虚函数的派生类,才能实例化对象。

接口类

        特殊的抽象类, 类中所有函数都是纯虚函数
实例如下:
/***
 *  C++ 多态
 *      动态多态
 *          虚函数/纯虚函数
 *          抽象类/接口类
 */

 #include <iostream>

 // 抽象类
 class Organism{
public:
    virtual void growup() = 0;
    virtual void eating(){
        std::cout << "eating everything" << std::endl;
    };
 };

// 基类 Animal 还是抽象类,无法实例化对象
class Animal : public Organism{
public:
    // 虚函数 speak()
    virtual void speak() {
        std::cout << "动物发出声音" << std::endl;
    }
};

// 派生类 Cat
class Cat : public Animal {
public:
    void growup(){
        std::cout<< "猫越长越大" << std::endl;
    }
    // 重写 speak() 函数
    void speak() override {
        std::cout << "喵~" << std::endl;
    }
};

// 派生类 Dog
class Dog : public Animal {
public:
    void growup(){
        std::cout<< "狗越长越大" << std::endl;
    }
    // 重写 speak() 函数
    void speak() override {
        std::cout << "汪!" << std::endl;
    }
};

int main() {
    // 创建 Cat 和 Dog 对象
    Cat cat;
    Dog dog;

    // 使用基类指针指向派生类对象
    Animal* animal1 = &cat;
    Animal* animal2 = &dog;

    // 调用虚函数
    animal1->speak(); // 输出:喵~
    animal2->speak(); // 输出:汪!
    animal1->growup(); // 输出:喵~
    animal2->growup(); // 输出:汪!

    animal1->Animal::speak(); // 输出:动物发出声音
    return 0;
}

五、多态的优点和应用场景

优点

  1. 灵活性和扩展性:运行期多态允许我们在不修改现有代码的情况下,轻松地添加新的派生类和功能
  2. 代码复用:通过使用基类的指针或引用,我们可以编写通用的代码来处理不同类型的对象,从而提高代码的复用性。例如,图形绘制程序中,我们可以编写一个通用的函数来计算不同图形的面积,而不需要为每种图形编写单独的函数。
  3. 可维护性:运行期多态使得代码的结构更加清晰,易于理解和维护。通过将不同的行为封装在不同的派生类中,我们可以避免代码的冗余和混乱。

应用场景

  1. 游戏开发:在游戏开发中,运行期多态可以用于实现不同角色的不同行为,如攻击、防御、移动等。例如,不同类型的角色(如战士、法师、刺客)可以继承自一个基类 Character,并分别重写 attack()defend()move() 等函数,以实现不同的行为。
  2. 图形处理:在图形处理中,运行期多态可以用于实现不同图形的不同绘制方法。例如,不同类型的图形(如圆形、矩形、三角形)可以继承自一个基类 Shape,并分别重写 draw() 函数,以实现不同的绘制方法。
  3. 数据库访问:在数据库访问中,运行期多态可以用于实现不同数据库的不同访问方法。例如,不同类型的数据库(如 MySQL、Oracle、SQLite)可以继承自一个基类 Database并分别重写 connect()query()insert() 等函数,以实现不同的访问方法。

六、总结

        运行期多态是 C++ 中一个非常强大和重要的特性,它通过虚函数和继承机制实现了在程序运行时根据实际对象的类型来决定执行哪个函数的功能。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2328685.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

饮食助力进行性核上性麻痹患者,提升生活质量

进行性核上性麻痹是一种少见的神经系统变性疾病&#xff0c;患者会出现姿势不稳、眼球运动障碍等症状。合理的饮食对于维持患者身体机能、延缓病情发展有重要意义。 高蛋白质食物是饮食结构的重要部分。像瘦肉、去皮禽肉、鱼类、豆类及其制品&#xff0c;还有低脂奶制品等&…

bun 版本管理工具 bum 安装与使用

在使用 node 的过程中&#xff0c;我们可能会因为版本更新或者不同项目的要求而频繁切换 node 版本&#xff0c;或者是希望使用更简单的方式安装不同版本的 node&#xff0c;这个时候我们一般会用到 nvm 或者类似的工具。 在我尝试使用 bun 的时候&#xff0c;安装前第一个想到…

木马学习记录

一句话木马是什么 一句话木马就是仅需要一行代码的木马&#xff0c;很简短且简单&#xff0c;木马的函数将会执行我们发送的命令 如何发送命令&#xff06;发送的命令如何执行? 有三种方式&#xff1a;GET&#xff0c;POST&#xff0c;COOKIE&#xff0c;一句话木马中用$_G…

决策树实战:用Python实现智能分类与预测

目录 一、环境准备 二、数据加载与探索 三、数据预处理 四、决策树模型构建 五、模型可视化&#xff08;生成决策树结构图&#xff09; 六、模型预测与评估 七、超参数调优&#xff08;网格搜索&#xff09; 八、关键知识点解析 九、完整项目开发流程 十、常见问题解…

Crond任务调度

今天我们来看看任务调度,假如我们正在睡觉,突然有个半夜两点的任务要你备份一下数据库,你怎么办&#xff1f;难道从被窝中爬起来吗&#xff1f;显然不合理,此时就需要我们定时任务调度程序了. 原理图&#xff1a; crontab 进行定时任务的调度 概述. 任务调度:是指系统在某个…

HTML5+CSS3+JS小实例:带滑动指示器的导航图标

实例:带滑动指示器的导航图标 技术栈:HTML+CSS+JS 效果: 源码: 【HTML】 <!DOCTYPE html> <html lang="zh-CN"> <head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, ini…

MINIQMT学习课程Day7

在上一篇&#xff0c;我们安装好xtquant&#xff0c;qmt以及python后&#xff0c;这一章&#xff0c;我们学习如何使用xtquant 本章学习&#xff0c;如何获取账号的资金使用状况。 首先&#xff0c;打开qmt&#xff0c;输入账号密码&#xff0c;选择独立交易。 进入交易界面&…

git clone 提示需要登录 github

我们在进行git的时候&#xff0c;可能会弹出让你登陆github的选项&#xff0c;这里我们介绍Token登陆的方法。 正常登陆你的Github 下拉找到 Developer settings按照如下步骤进行操作 填写相关信息&#xff0c;勾选对应选项 返回就能看到token已经被生成&#xff0c;可以使…

4.2-3 fiddler抓取手机接口

安卓&#xff1a; 长按手机连接的WiFi&#xff0c;点击修改网络 把代理改成手动&#xff0c;服务器主机选择自己电脑的IP地址&#xff0c;端口号为8888&#xff08;在dos窗口输入ipconfig查询IP地址&#xff0c;为ipv4&#xff09; 打开手机浏览器&#xff0c;输入http://自己…

Nacos注册中心AP模式核心源码分析(单机模式)

文章目录 概述一、客户端启动主线流程源码分析1.1、客户端与Spring Boot整合1.2、注册实例&#xff08;服务注册&#xff09;1.3、发送心跳1.4、拉取服务端实例列表&#xff08;服务发现&#xff09; 二、服务端接收请求主线流程源码分析2.1、接收注册请求2.1.1、初始化注册表2…

【进收藏夹吃灰】机器学习学习指南

博客标题URL【机器学习】线性回归&#xff08;506字&#xff09;https://blog.csdn.net/from__2025_03_16/article/details/146303423

【Web 服务器】的工作原理

&#x1f310; Web 服务器的工作原理 Web 服务器的主要作用是 接收客户端请求&#xff08;通常是浏览器发出的 HTTP/HTTPS 请求&#xff09;&#xff0c;处理请求&#xff0c;并返回相应的数据&#xff08;如网页、图片、API 响应等&#xff09;。 &#x1f4cc; 工作流程 1️…

【Cursor】设置语言

Ctrl Shift P 搜索 configure display language选择“中文-简体”

k8s 1.30 安装ingress-nginx

一、下载 # wget https://mirrors.chenby.cn/https://raw.githubusercontent.com/kubernetes/ingress-nginx/main/deploy/static/provider/cloud/deploy.yaml 二、过滤镜像 修改 三、部署 四、检查&#xff1a; 五、扩充副本数 # kubectl scale --replicas3 deployment/ingr…

很简单 的 将字幕生成视频的 方法

一、一键将字幕生成视频的 方法 1、下载任性动图 10.7 以上版本 2、设置背景 1&#xff09;背景大小 拉伸背景到合适大小&#xff0c;或者选择右侧比例 2&#xff09;、直接空背景&#xff0c;设置背景颜色等详细信息 3&#xff09;、或者 复制或者突然图片做背景 3、设置文…

OpenCv(二)——边界填充、阈值处理

目录 一、边界填充 &#xff08;1&#xff09;constant边界填充&#xff0c;填充指定宽度的像素 &#xff08;2&#xff09;REFLECT镜像边界填充 &#xff08;3&#xff09;REFLECT_101镜像边界填充改进 (4) REPLICATE使用最边界的像素值代替 (5)WRAP上下左右边依次替换 二…

理解OSPF Stub区域和各类LSA特点

之前学习到OSPF特殊区域和各类类型LSA的分析后&#xff0c;一直很混乱&#xff0c;在网上也难找到详细的解释&#xff0c;在看了 HCNP书本内容后&#xff0c;对这块类容理解更加清晰&#xff0c;本次内容&#xff0c;我们使用实验示例&#xff0c;来对OSPF特殊区域和各 类型LSA…

CAN/FD CAN总线配置 最新详解 包含理论+实战(附带源码)

看前须知&#xff1a;本篇文章不会说太多理论性的内容&#xff08;重点在理论结合实践&#xff09;&#xff0c;顾及实操&#xff0c;应用&#xff0c;一切理论内容支撑都是为了后续实际操作进行铺垫&#xff0c;重点在于读者可以看完文章应用。&#xff08;也为节约读者时间&a…

MoE Align Sort在医院AI医疗领域的前景分析(代码版)

MoE Align & Sort技术通过优化混合专家模型(MoE)的路由与计算流程,在医疗数据处理、模型推理效率及多模态任务协同中展现出显著优势,其技术价值与应用意义从以下三方面展开分析: 一、方向分析 1、提升医疗数据处理效率 在医疗场景中,多模态数据(如医学影像、文本…

【已解决】Webstorm 每次使用 git pull/push 都要输入令牌/密码登录

解决办法&#xff1a;勾上【使用凭据帮助程序】&#xff08;英文&#xff1a;Use credential helper&#xff09;