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

C++基础系列【14】继承与多态

博主介绍:程序喵大人

  • 35- 资深C/C++/Rust/Android/iOS客户端开发
  • 10年大厂工作经验
  • 嵌入式/人工智能/自动驾驶/音视频/游戏开发入门级选手
  • 《C++20高级编程》《C++23高级编程》等多本书籍著译者
  • 更多原创精品文章,首发gzh,见文末
  • 👇👇记得订阅专栏,以防走丢👇👇
    😃C++基础系列专栏
    😃C语言基础系列

本文主要介绍下多态的概念。

继承与抽象类

多态是面向对象的核心知识点,在C++中意味着调用对象成员函数时,会根据对象的真实类型来执行不同的函数,从而产生不同的行为。

  • 比如同样是人,不同人的声音不相同。
  • 比如同样是公司,不同公司的经营业务也不同。

这就可以就多态来解释。

怎么实现多态,看这段代码,先定义一个People类:

class People {
public:
virtual void Speak() { std::cout << "People Speak \n"; }
};

注意这里面的函数使用了virtual修饰,用virtual修饰的函数表示虚函数,带虚函数的类可以称之为父类,有父类自然可以派生出子类,子类可以覆盖父类的行为。

这里再定义两个类,一个男人类,一个女人类

class MalePeople : public People {
public:
void Speak() { std::cout << "MalePeople Speak \n"; }
};class FemalePeople : public People {
public:
void Speak() { std::cout << "FemalePeople Speak \n"; }
};

MalePeopleFemalePeople使用了冒号,表示继承,冒号后面的public表示继承的权限。

所以上面的代码的含义是:

MalePeoplepublic权限继承了People,并覆盖父类PeopleSpeak行为。

FemalePeoplepublic权限继承了People,并覆盖父类PeopleSpeak行为。

再看一段使用多态的代码:

int main() {People *p1 = new People();People *p2 = static_cast<People *>(new MalePeople());People *p3 = static_cast<People *>(new FemalePeople());p1->Speak(); // People Speakp2->Speak(); // MalePeople Speakp3->Speak(); // FemalePeople Speakdelete p3;delete p2;delete p1;
}

p1、p2、p3都是People的实例,但是通过他们的实例调用相同的函数却产生了不同的行为,这就是多态

注意两点,想要实现上述的多态行为:

  • 父类相应的函数一定要使用virtual修饰
  • 一定要父类的指针或引用指向子类对象

继承权限

共有三种继承权限:

public继承

  • 父类中所有public成员在子类中为public属性
  • 父类中所有protected成员在子类中为protected属性
  • 父类中所有private成员在子类中不可访问

protected继承

  • 父类中所有public成员在子类中为protected属性
  • 父类中所有protected成员在子类中为protected属性
  • 父类中所有private成员在子类中不可访问

private继承

  • 父类中所有public成员在子类中为private属性
  • 父类中所有protected成员在子类中为private属性
  • 父类中所有private成员在子类中不可访问

大体可以理解为:

  • 父类成员在子类中的访问权限不会高于指定的继承权限。
  • 父类中的private成员在子类中使用不可访问。

然而平时开发过程中一般都会使用public继承,其他的继承方式很少。

纯虚函数

在C++中,还有个纯虚函数的概念,就是在virtual修饰的基础上加个=0,比如:

class People {
public:
virtual void Speak() = 0;
};

这里的Speak就是纯虚函数,含有纯虚函数的类叫抽象类,同时规定抽象类不允许被实例化,只能通过子类实例化,举例:

int main() {People *p1 = new People(); // compile errorPeople *p2 = static_cast<People *>(new MalePeople());People *p3 = static_cast<People *>(new FemalePeople());
}

多继承

就是子类继承了多个父类,比如一个男子篮球运动员,那就可以定义两个父类,一个MalePeople类,一个BasketballPlayer类,那如果想要定义男子篮球运动员类,可以定义一个MaleBasketballPlayer类,继承MalePeopleBaskeballPlayer,代码如下:

class MalePeople {
public:
void Speak() { std::cout << "MalePeople Speak \n"; }
};class BasketBallPlayer {
public:
void Play() { std::cout << "Play Basketball \n"; }
};class MaleBasketBallPlayer : public MalePeople, public BasketBallPlayer {};

和单继承方式差不多,只是用相同的语法在后面再派生多个即可。

虚继承

在这里插入图片描述

普通的继承就是非虚继承,如图, 非虚继承时,显然D会继承两次A,内部就会存储两份A的数据浪费空间,而且还有二义性,D调用A的方法时,由于有两个A,究竟时调用哪个A的方法呢,编译器也不知道,就会报错,所以有了虚继承,解决了空间浪费以及二义性问题。

在这里插入图片描述

在虚拟继承下,只有一个共享的基类子对象被继承,而无论该基类在派生层次中出现多少次。共享的基类子对象被称为虚基类。在虚继承下,基类子对象的复制及由此而引起的二义性都被消除了。

如何使用虚继承?

在继承的时候使用virtual关键字,代码如下:

struct Base {
virtual void Func() { printf("Base Func\n"); }
};struct BaseA : virtual public Base {
virtual void Func() { printf("BaseA Func\n"); }
};struct BaseB : virtual public Base {
virtual void Func() { printf("BaseB Func\n"); }
};struct Derive : public BaseB, public BaseA {
void Func() override { printf("Derive Func \n"); }
};

注意,为了易于观察,上面所有的父类都没有定义析构函数,正常父类的析构函数一定要设置成virtual

练习

  • 多态只有这一种方式吗?
  • 为什么一定要通过指针或引用方式才能达到多态的目的?
  • 为什么析构函数一定要设置成virtual?
  • 构造函数可以为虚函数吗?
  • 多态的原理是怎么样的?
  • 不同继承方式下,类对象的布局是什么结构?

码字不易,欢迎大家点赞关注评论,谢谢!


C++训练营

专为校招、社招3年工作经验的同学打造的1V1 C++训练营,量身定制学习计划、每日代码review,简历优化,面试辅导,已帮助多名学员获得offer!训练营介绍

http://www.lryc.cn/news/538035.html

相关文章:

  • DeepSeek-R1 大模型本地部署指南
  • 在conda环境下,安装Pytorch和CUDA
  • Java里int和Integer的区别?
  • 【第13章:自监督学习与少样本学习—13.4 自监督学习与少样本学习的未来研究方向与挑战】
  • 【NLP】文本预处理
  • deepseek r1从零搭建本地知识库10:嵌入模型和知识库建设
  • Linux-文件IO
  • 3d pose 学习笔记2025
  • LC-随机链表的复制、排序链表、合并K个升序链表、LRU缓存
  • 静态页面在安卓端可以正常显示,但是在ios打开这个页面就需要刷新才能显示全图片
  • 四元数如何用于 3D 旋转(代替欧拉角和旋转矩阵)【ESP32指向鼠标】
  • JavaScript 内置对象-日期对象
  • 本地大模型编程实战(19)RAG(Retrieval Augmented Generation,检索增强生成)(3)
  • DeepSeek与ChatGPT:AI语言模型的全面对决
  • 2024年年终总结
  • 利用 Valgrind 检测 C++ 内存泄露
  • Python中的HTTP客户端库:httpx与request | python小知识
  • 【Python】Python入门基础——环境搭建
  • 2025 pwn_A_childs_dream
  • 面试题整理:操作系统
  • 构建未来教育的基石:智慧校园与信息的重要性
  • C# 控制台相关 API 与随机数API
  • 【踩坑】⭐️MyBatis的Mapper接口中不建议使用重载方法
  • CSS Grid 网格布局,以及 Flexbox 弹性盒布局模型,它们的适用场景是什么?
  • HDFS体系结构
  • AI大模型的技术突破与传媒行业变革
  • vscode/cursor+godot C#中使用socketIO
  • 分段线性插值
  • 制作一个项目用于研究elementUI的源码
  • [AI]从零开始的llama.cpp部署与DeepSeek格式转换、量化、运行教程