C++ 多态:面向对象的动态行为核心机制

C++ 多态:面向对象的动态行为核心机制

C++ 多态:面向对象的动态行为核心机制

在这里插入图片描述

💡 学习目标:掌握多态的概念与分类,理解虚函数的作用原理,能够熟练使用多态实现程序的动态行为扩展。
💡 学习重点:静态多态与动态多态的区别、虚函数的定义与使用、纯虚函数与抽象类、多态的实战应用场景。

一、多态的概念与分类

结论:多态是 C++ 面向对象三大特性之一,指同一行为在不同对象上表现出不同的形态,核心是“一个接口,多种实现”。

多态主要分为两大类,二者的实现原理和触发时机截然不同:

  1. 静态多态:编译阶段确定调用关系,也叫编译时多态,实现方式包括函数重载运算符重载
  2. 动态多态:运行阶段确定调用关系,也叫运行时多态,实现方式是虚函数 + 基类指针/引用

生活中的多态示例:同样是“动物叫”这个行为,猫的叫声是“喵喵喵”,狗的叫声是“汪汪汪”,不同动物对象表现出不同的行为形态。

二、静态多态:编译时确定的多态性

💡 静态多态的调用关系在编译阶段就已确定,编译器会根据参数列表的差异匹配对应的函数。

2.1 函数重载实现静态多态

这是最常见的静态多态形式,同一作用域内的同名函数,通过参数的类型或数量区分。

#include<iostream>usingnamespace std;// 静态多态:函数重载voidprint(int a){ cout <<"整数:"<< a << endl;}voidprint(double b){ cout <<"浮点数:"<< b << endl;}voidprint(string c){ cout <<"字符串:"<< c << endl;}intmain(){// 编译阶段确定调用哪个 print 函数print(10);print(3.14);print("C++ Static Polymorphism");return0;}

2.2 运算符重载实现静态多态

通过重载运算符,让自定义类型支持内置运算符的操作,本质也是编译时多态。

#include<iostream>usingnamespace std;classPoint{public:int x, y;Point(int x =0,int y =0):x(x),y(y){}// 重载 + 运算符,实现点的坐标相加 Point operator+(const Point& p){returnPoint(this->x + p.x,this->y + p.y);}};intmain(){ Point p1(1,2),p2(3,4);// 编译阶段确定调用重载的 + 运算符 Point p3 = p1 + p2; cout <<"p3.x = "<< p3.x <<", p3.y = "<< p3.y << endl;return0;}

三、动态多态:运行时确定的多态性

💡 动态多态是面向对象编程的核心,其关键是虚函数基类指针/引用指向派生类对象,调用关系在程序运行时才确定。

3.1 动态多态的实现条件

实现动态多态必须同时满足三个条件:

  1. 存在继承关系,且通常是公有继承
  2. 基类中定义虚函数,派生类重写该虚函数
  3. 使用基类的指针或引用指向派生类对象

3.2 虚函数的定义与使用

虚函数的定义语法是在基类的成员函数前加 virtual 关键字,派生类重写时可以省略 virtual,但建议保留以增强可读性。

3.2.1 基础案例:动物叫的多态实现
#include<iostream>#include<string>usingnamespace std;// 基类:动物classAnimal{public:// 虚函数:动物叫virtualvoidbark(){ cout <<"动物发出叫声"<< endl;}};// 派生类:猫classCat:publicAnimal{public:// 重写基类的虚函数voidbark()override{// override 关键字显式声明重写,建议添加 cout <<"猫:喵喵喵"<< endl;}};// 派生类:狗classDog:publicAnimal{public:// 重写基类的虚函数voidbark()override{ cout <<"狗:汪汪汪"<< endl;}};intmain(){// 基类指针指向派生类对象 Animal *animal1 =newCat(); Animal *animal2 =newDog();// 运行时确定调用哪个类的 bark 函数 animal1->bark();// 输出猫的叫声 animal2->bark();// 输出狗的叫声// 释放内存delete animal1;delete animal2;return0;}
3.2.2 运行结果
猫:喵喵喵 狗:汪汪汪 

⚠️ 注意事项

  1. override 关键字用于显式标记派生类对基类虚函数的重写,编译器会检查重写的合法性,建议添加。
  2. 如果基类指针指向基类对象,则调用基类的虚函数;指向派生类对象,则调用派生类重写的函数。
  3. 虚函数的重写要求函数名、参数列表、返回值类型完全一致,否则会被视为派生类的新函数,而非重写。

3.3 虚析构函数:解决派生类资源泄漏问题

当基类指针指向派生类对象并通过 delete 释放时,如果基类析构函数不是虚函数,会导致派生类的析构函数无法被调用,从而引发内存泄漏。

解决方案:将基类的析构函数声明为虚析构函数

3.3.1 问题代码演示(非虚析构)
#include<iostream>usingnamespace std;classBase{public:Base(){ cout <<"Base 构造函数"<< endl;}~Base(){ cout <<"Base 析构函数"<< endl;}// 非虚析构};classDerived:publicBase{public:Derived(){ cout <<"Derived 构造函数"<< endl;}~Derived(){ cout <<"Derived 析构函数"<< endl;}// 无法被调用};intmain(){ Base *p =newDerived();delete p;// 仅调用基类析构函数,派生类析构未调用return0;}
3.3.2 运行结果(存在问题)
Base 构造函数 Derived 构造函数 Base 析构函数 
3.3.3 解决代码(虚析构函数)
#include<iostream>usingnamespace std;classBase{public:Base(){ cout <<"Base 构造函数"<< endl;}virtual~Base(){ cout <<"Base 析构函数"<< endl;}// 虚析构函数};classDerived:publicBase{public:Derived(){ cout <<"Derived 构造函数"<< endl;}~Derived()override{ cout <<"Derived 析构函数"<< endl;}// 重写虚析构};intmain(){ Base *p =newDerived();delete p;// 先调用派生类析构,再调用基类析构return0;}
3.3.4 运行结果(正确释放)
Base 构造函数 Derived 构造函数 Derived 析构函数 Base 析构函数 

四、纯虚函数与抽象类

💡 纯虚函数是没有函数体的虚函数,包含纯虚函数的类称为抽象类,抽象类无法实例化对象,只能作为基类被继承。

4.1 纯虚函数的定义语法

virtual 返回值类型 函数名(参数列表)=0;

4.2 抽象类的特性

  1. 抽象类不能创建对象,只能定义指针或引用。
  2. 派生类必须重写抽象类的所有纯虚函数,否则派生类也会成为抽象类。
  3. 抽象类的核心作用是定义接口规范,强制派生类实现具体功能。

4.3 代码演示:图形面积计算的抽象类

#include<iostream>usingnamespace std;// 抽象类:图形classShape{public:// 纯虚函数:计算面积virtualdoublegetArea()=0;// 纯虚函数:计算周长virtualdoublegetPerimeter()=0;};// 派生类:矩形classRectangle:publicShape{private:double width, height;public:Rectangle(double w,double h):width(w),height(h){}// 必须重写所有纯虚函数doublegetArea()override{return width * height;}doublegetPerimeter()override{return2*(width + height);}};// 派生类:圆形classCircle:publicShape{private:double radius;constdouble PI =3.1415926;public:Circle(double r):radius(r){}doublegetArea()override{return PI * radius * radius;}doublegetPerimeter()override{return2* PI * radius;}};// 通用函数:打印图形信息voidprintShapeInfo(Shape *shape){ cout <<"面积:"<< shape->getArea()<< endl; cout <<"周长:"<< shape->getPerimeter()<< endl;}intmain(){// 抽象类不能实例化对象// Shape s; // 编译错误 Shape *rect =newRectangle(5,3); Shape *circle =newCircle(4); cout <<"矩形信息:"<< endl;printShapeInfo(rect); cout <<"----------------"<< endl; cout <<"圆形信息:"<< endl;printShapeInfo(circle);delete rect;delete circle;return0;}

4.4 运行结果

矩形信息: 面积:15 周长:16 ---------------- 圆形信息: 面积:50.2655 周长:25.1327 

五、多态的实战案例:计算器的动态扩展

💡 需求:设计一个支持多种运算的计算器,利用多态特性,让计算器可以动态扩展新的运算类型,无需修改原有代码(符合开闭原则)。

5.1 需求分析

  1. 定义抽象基类 Operation,包含纯虚函数 calculate,用于计算结果。
  2. 派生类分别实现加法 Add、减法 Sub、乘法 Mul、除法 Div
  3. 设计计算器类 Calculator,接收 Operation 指针,调用计算方法。
  4. 新增运算类型时,只需新增派生类,无需修改计算器核心代码。

5.2 完整代码实现

#include<iostream>#include<stdexcept>usingnamespace std;// 抽象基类:运算classOperation{public:double num1, num2;voidsetNum(double n1,double n2){ num1 = n1; num2 = n2;}// 纯虚函数:计算virtualdoublecalculate()=0;};// 加法运算classAdd:publicOperation{public:doublecalculate()override{return num1 + num2;}};// 减法运算classSub:publicOperation{public:doublecalculate()override{return num1 - num2;}};// 乘法运算classMul:publicOperation{public:doublecalculate()override{return num1 * num2;}};// 除法运算classDiv:publicOperation{public:doublecalculate()override{if(num2 ==0){throwinvalid_argument("除数不能为 0");}return num1 / num2;}};// 计算器类classCalculator{private: Operation *op;public:Calculator(Operation *operation):op(operation){}doublecompute(double n1,double n2){ op->setNum(n1, n2);return op->calculate();}~Calculator(){delete op;}};intmain(){try{// 加法计算 Calculator addCalc(newAdd()); cout <<"10 + 5 = "<< addCalc.compute(10,5)<< endl;// 减法计算 Calculator subCalc(newSub()); cout <<"10 - 5 = "<< subCalc.compute(10,5)<< endl;// 乘法计算 Calculator mulCalc(newMul()); cout <<"10 * 5 = "<< mulCalc.compute(10,5)<< endl;// 除法计算 Calculator divCalc(newDiv()); cout <<"10 / 5 = "<< divCalc.compute(10,5)<< endl;// 测试除数为 0 cout <<"10 / 0 = "<< divCalc.compute(10,0)<< endl;}catch(const exception& e){ cout <<"错误:"<< e.what()<< endl;}return0;}

5.3 运行结果

10 + 5 = 15 10 - 5 = 5 10 * 5 = 50 10 / 5 = 2 错误:除数不能为 0 

六、多态的核心原理:虚函数表

💡 C++ 动态多态的底层实现依赖虚函数表(vtable)虚函数指针(vptr),理解其原理能帮助我们更好地使用多态。

6.1 虚函数表的工作机制

  1. 当类中包含虚函数时,编译器会为该类生成一个虚函数表,表中存储的是虚函数的地址。
  2. 每个对象的内存布局中,会有一个隐藏的虚函数指针 vptr,指向所属类的虚函数表。
  3. 当基类指针指向派生类对象时,vptr 会指向派生类的虚函数表。
  4. 程序运行时,通过 vptr 找到虚函数表,再根据表中的地址调用对应的函数,从而实现动态多态。

6.2 核心特点

  • 虚函数表属于,所有对象共享同一个虚函数表,节省内存空间。
  • 虚函数指针属于对象,每个对象都有自己的 vptr,指向对应类的虚函数表。
  • 派生类的虚函数表会继承基类的虚函数表,并重写被覆盖的虚函数地址。

七、本章总结

✅ 多态分为静态多态和动态多态,静态多态编译时确定,动态多态运行时确定。
✅ 动态多态的实现条件是:继承关系 + 虚函数重写 + 基类指针/引用指向派生类对象。
✅ 虚析构函数可以解决基类指针释放派生类对象时的资源泄漏问题。
✅ 抽象类包含纯虚函数,无法实例化,用于定义接口规范,强制派生类实现具体功能。
✅ 多态的核心优势是代码复用、功能扩展、符合开闭原则,是大型项目设计的核心机制。

Read more

【离散化 线段树 二分查找】3661可以被机器人摧毁的最大墙壁数目|2525

【离散化 线段树 二分查找】3661可以被机器人摧毁的最大墙壁数目|2525

本文涉及知识点 【C++】树状数组的使用、原理、封装类、样例 C++线段树 C++二分查找 3661. 可以被机器人摧毁的最大墙壁数目 一条无限长的直线上分布着一些机器人和墙壁。给你整数数组 robots ,distance 和 walls: robots[i] 是第 i 个机器人的位置。 distance[i] 是第 i 个机器人的子弹可以行进的 最大 距离。 walls[j] 是第 j 堵墙的位置。 每个机器人有 一颗 子弹,可以向左或向右发射,最远距离为 distance[i] 米。 子弹会摧毁其射程内路径上的每一堵墙。机器人是固定的障碍物:如果子弹在到达墙壁前击中另一个机器人,它会 立即 在该机器人处停止,无法继续前进。

By Ne0inhk

FPGA摄像头到屏幕完整链路:从OV5640采集到HDMI实时显示(附完整工程代码)

🎬 FPGA摄像头到屏幕完整链路:从OV5640采集到HDMI实时显示(附完整工程代码) 📚 目录导航 文章目录 * 🎬 FPGA摄像头到屏幕完整链路:从OV5640采集到HDMI实时显示(附完整工程代码) * 📚 目录导航 * 概述 * 一、摄像头采集显示系统架构 * 1.1 系统整体框架 * 1.2 核心模块功能 * 1.3 数据流向与时序 * 二、OV5640摄像头基础 * 2.1 OV5640摄像头简介 * 2.2 OV5640引脚定义与功能 * 2.3 DVP接口时序详解 * 2.4 SCCB配置协议 * 2.5 OV5640初始化配置 * 三、图像采集模块设计 * 3.1 DVP采集模块架构 * 3.2 行列计数器设计 * 3.3 数据格式转换 * 3.

By Ne0inhk
近五年体内微/纳米机器人赋能肿瘤精准治疗综述:以 GBM 为重点

近五年体内微/纳米机器人赋能肿瘤精准治疗综述:以 GBM 为重点

摘要 实体瘤治疗长期受制于递送效率低、肿瘤组织渗透不足以及免疫抑制与耐药等问题。传统纳米药物多依赖被动累积与扩散,难以在肿瘤内部形成均匀有效的药物浓度分布。2021–2025 年,体内微/纳米机器人(包括外场驱动微型机器人、自驱动纳米马达以及生物混合机器人)围绕“运动能力”形成了三条相互收敛的技术路线: 其一,通过磁驱、声驱、光/化学自驱等方式实现运动增强递药与深层渗透,将治疗从“被动到达”推进到“主动进入”; 其二,与免疫治疗深度融合,实现原位免疫唤醒与肿瘤微环境重塑; 其三,针对胶质母细胞瘤(glioblastoma, GBM)等难治肿瘤,研究趋势转向“跨屏障递送(BBB/BBTB)+ 成像/外场闭环操控 + 时空可控释放”的系统工程。 本文围绕“运动—分布—疗效”的因果链条,总结 2021–2025 年代表性研究与关键评价指标,讨论临床转化所需的安全性、

By Ne0inhk
宇树G1机器人强化学习训练完整实战教程

宇树G1机器人强化学习训练完整实战教程

0. 前言 人形机器人的运动控制一直是机器人领域的重要挑战,而强化学习为解决这一问题提供了强有力的工具。本教程将基于宇树G1人形机器人,从基础的强化学习环境搭建开始,逐步深入到高自由度模型的训练配置、奖励函数设计与优化,最终实现复杂动作的训练控制。作者看到一个很棒的系列,所以针对性的对文章内容进行了整理和二次理解,方便大家更好的阅读《不同自由度的宇树G1机器人强化学习训练配置及运行实战 + RSL-RL代码库问题修复》、《宇树G1机器人强化学习训练奖励函数代码架构 + 创建新的奖励函数(1)》、《RL指标分析与看板应用 — 宇树G1机器人高自由度模型强化学习训练实战(3)》、《调参解析 — 宇树G1机器人高自由度模型强化学习训练实战(4)》、《舞蹈训练?手撕奖励函数 — 宇树G1机器人高自由度模型强化学习训练实战(5)》。 1. 强化学习训练环境配置 1.1 基础环境搭建 宇树机器人的强化学习训练基于Isaac Gym物理仿真环境和RSL-RL强化学习框架。首先需要确保这两个核心组件正确安装和配置。 在开始训练之前,我们通过简单的命令来启动12自由度G1机器人的基础训练:

By Ne0inhk