【C++深度探索】继承机制详解(二)

hello hello~ ,这里是大耳朵土土垚~💖💖 ,欢迎大家点赞🥳🥳关注💥💥收藏🌹🌹🌹
在这里插入图片描述

💥个人主页:大耳朵土土垚的博客
💥 所属专栏:C++入门至进阶

这里将会不定期更新有关C++的内容,希望大家多多点赞关注收藏💖💖

目录

  • 1.继承与友元
  • 2.继承与静态成员
  • 3.单继承与多继承
  • 4.菱形继承与虚拟继承
  • 5.虚拟继承
  • 6.继承和组合
  • 7.笔试面试题
  • 8.结语

1.继承与友元

基类的友元关系不能被子类继承,也就是说基类友元不能访问子类私有和保护成员

在C++的继承中,友元函数并不具有继承的特性。当一个类继承另一个类时,它只会继承基类的成员函数和数据成员,而不会继承基类中声明的友元函数。

例如:

class Student;//声明
//基类
class Person
{
public:friend void Display(const Person& p, const Student& s);//友元函数
protected:string _name; // 姓名
};
//子类
class Student : public Person
{
protected:int _stuNum; // 学号
};void Display(const Person& p, const Student& s)
{cout << p._name << endl;//可以访问基类的成员cout << s._stuNum << endl;//不可以访问子类的成员
}
int main()
{Person p;Student s;Display(p, s);return 0;
}

结果如下:

在这里插入图片描述

因此,友元函数不能被继承。每个类都需要单独定义其自己的友元函数。

2.继承与静态成员

基类定义了static静态成员,则整个继承体系里面只有一个这样的成员。无论派生出多少个子类,都只有一个static成员实例。

静态成员是属于类而不是对象的成员。它们与类相关联,而不是与类的每个对象相关联。静态成员可以是静态变量和静态函数。

例如:

class Person
{
public:Person() { ++_count; }	//默认构造,每次创建对象自动调用
protected:string _name; // 姓名
public:static int _count; // 静态成员变量_count,统计人的个数
};int Person::_count = 0;//初始化静态成员变量class Student : public Person
{
protected:int _stuNum; // 学号
};class Graduate : public Student
{
protected:string _seminarCourse; // 研究科目
};void TestPerson()
{Student s1;Student s2;Student s3;Graduate s4;cout << " 人数 :" << Person::_count << endl;Student::_count = 0;cout << " 人数 :" << Person::_count << endl;
}

结果如下:

在这里插入图片描述

可以看出类的静态成员与普通成员不同,它独立于各个对象,存放在静态存储区,即使基类有多个子类,静态成员也不会被拷贝多次,是它们公共使用的。

3.单继承与多继承

  • 单继承:一个子类只有一个直接父类时称这个继承关系为单继承
    如下图所示:
    在这里插入图片描述

上图中,Student类只有一个直接父类Person,PostGraduate类也只有一个直接父类Student

  • 多继承:一个子类有两个或以上直接父类时称这个继承关系为多继承
    如下图所示:
    在这里插入图片描述

上图中,Assistant类有两个直接父类Student类和Teacher类

4.菱形继承与虚拟继承

  • 菱形继承:菱形继承是多继承的一种特殊情况

菱形继承是指在C++中使用多重继承时,出现了多个派生类继承同一个基类,而最终有一个类同时继承了这些派生类,形成了一个菱形的继承结构。

例如:

//菱形继承
class Person
{
public:string _name; // 姓名
};
class Student : public Person
{
protected:int _num; //学号
};
class Teacher : public Person
{
protected:int _id; // 职工编号
};
class Assistant : public Student, public Teacher
{
protected:string _majorCourse; // 主修课程
};

如下图所示:
在这里插入图片描述

Assistant类有两个直接父类Student类和Teacher类,而这两个类又都继承自Person类

当然不是所有的菱形继承都是一个标准的菱形结构,也可能是别的更复杂的结构,只要出现了多个派生类继承同一个基类,而最终有一个类同时继承了这些派生类,就形成了一个菱形的继承结构。

菱形继承带来的问题

菱形继承可以带来一些问题,主要是关于数据冗余和二义性。

  • 数据冗余
    上图中Student类和Teacher类继承自同一父类Person,有相同的数据成员,那么在Assistant类中就会有两份相同的数据成员,会引起数据冗余
    在这里插入图片描述

  • 二义性
    如果Student类和Teacher类都有相同的成员,那么当在Assistant类中调用这个成员时,会出现二义性。编译器不知道应该调用Student类中的成员还是Teacher类中的成员。
    例如:

void Test()
{// 这样会有二义性无法明确知道访问的是哪一个Assistant a;a._name = "peter";// 需要显示指定访问哪个父类的成员可以解决二义性问题,但是数据冗余问题无法解决a.Student::_name = "xxx";a.Teacher::_name = "yyy";
}

在这里插入图片描述

5.虚拟继承

为了解决这些问题,C++提供了虚继承的机制。通过在中间派生类Student和Teacher的继承声明中加上关键字virtual,可以实现虚继承。虚继承可以解决菱形继承带来的二义性问题,确保在最终派生类Assistant中只有一份数据成员和函数。需要注意的是,虚拟继承不要在其他地方去使用。

例如:

class Person
{
public:string _name; // 姓名
};//虚拟继承
class Student : virtual public Person
{
protected:int _num; //学号
};
//虚拟继承
class Teacher : virtual public Person
{
protected:int _id; // 职工编号
};class Assistant : public Student, public Teacher
{
protected:string _majorCourse; // 主修课程
};void Test()
{Assistant a;a._name = "peter";
}

在这里插入图片描述

我们看到这时再通过对象a去访问_name成员不会出现二义性,并且当给_name赋值时,发现在监视中所继承的_name都改变,这说明Assistant类中只有一份数据成员和函数,不会出现数据冗余

关于菱形虚拟继承的原理解释涉及到虚拟基类表与偏移量,大家有兴趣可以查看资料,这里就不过多介绍。

有了多继承,就存在菱形继承,有了菱形继承就有菱形虚拟继承,底层实现就很复杂。所以一般不建议设计出多继承,一定不要设计出菱形继承。否则在复杂度及性能上都有问题。

6.继承和组合

  • 组合是一种对象关系,一个类可以包含其他类的对象作为其成员变量。这种关系不是通过继承来实现,而是通过在一个类中创建另一个类的对象来实现。
  • 而继承是一种面向对象编程中的机制,允许一个类(称为派生类或子类)从另一个类(称为基类或父类)继承属性和行为。通过继承,子类可以继承父类的特征和功能,并且可以添加或修改自己的特性和功能。

public继承是一种is-a的关系。也就是说每个派生类对象都是一个基类对象。
例如:

class Person
{
public:string _name; // 姓名
};
class Student : public Person
{
protected:int _num; //学号
};Student s;

上述代码中,Student对象s是一个Person类的对象;放在现实生活中就是学生是一个人

组合是一种has-a的关系。假设B组合了A,每个B对象中都有一个A对象。

class A {
//...
};class B {A a;//...
};
  • 在继承方式中,基类的内部细节对子类可见 。继承一定程度破坏了基类的封装,基类的改变,对派生类有很大的影响。派生类和基类间的依赖关系很强,耦合度高。在继承方式中,基类的内部细节对子类可见 。继承一定程度破坏了基类的封装,基类的改变,对派生类有很大的影响。派生类和基类间的依赖关系很强,耦合度高
  • 对象组合要求被组合的对象具有良好定义的接口。因为对象的内部细节是不可见的,组合类之间没有很强的依赖关系,耦合度低。优先使用对象组合有助于你保持每个类被封装。

实际尽量多去用组合。组合的耦合度低,代码维护性好。不过继承也有用武之地的,有些关系就适合继承那就用继承,另外要实现多态,也必须要继承。类之间的关系可以用继承,可以用组合,就用组合。

7.笔试面试题

  1. 什么是菱形继承?菱形继承的问题是什么?
    ①菱形继承是多继承的一种特殊情况。菱形继承是指在C++中使用多重继承时,出现了多个派生类继承同一个基类,而最终有一个类同时继承了这些派生类,形成了一个菱形的继承结构。
    ②数据冗余和二义性。

  2. 什么是菱形虚拟继承?如何解决数据冗余和二义性的?
    ①为了解决数据冗余和二义性问题,C++提供了虚继承的机制。通过在中间派生类的继承声明中加上关键字virtual,将共同继承的基类标记为虚拟继承。
    ②虚拟继承确保在最终派生类中只有一份数据成员和函数,解决了数据冗余与二义性问题。

  3. 继承和组合的区别?什么时候用继承?什么时候用组合?
    ①public继承是一种is-a的关系。也就是说每个派生类对象都是一个基类对象。组合是一种has-a的关系。假设B组合了A,每个B对象中都有一个A对象。
    ②选择继承还是组合取决于具体的需求和设计问题。一般来说,当两个类之间存在明确的层次关系,并且子类是父类的一种特例时,可以使用继承。但是,继承可能导致类之间的耦合度较高,代码结构较为复杂,因此需要慎重使用。而当两个类之间没有明确的层次关系,或者需要构建复杂对象时,可以使用组合。组合可以帮助代码实现更灵活的结构,并且可以随时替换或扩展不同的组件。另外要实现多态,也必须要继承。

8.结语

在C++中,继承支持单继承与多继承,对于多继承中又包含菱形继承,如果实现了菱形继承就必须实现虚拟继承来解决数据冗余与二义性的问题,所以在实践中我们要尽量减少使用菱形继承,此外对于继承与组合的区别也需要好好掌握一下,以上就是今天所有的内容啦~ 完结撒花~ 🥳🎉🎉

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

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

相关文章

如何把已经上传到gitlab的代码或者文件夹从git上删掉

有小伙伴不小心把缓存文件上传到了git&#xff0c;跑来问我&#xff0c;要怎么把这些文件给删掉&#xff0c;这里一共有两种方式&#xff0c; 先说第一种&#xff0c;通过命令删除&#xff0c;终端进入存在这个缓存文件的目录&#xff0c;执行命令ls&#xff0c;可以看到确实有…

从零开始搭建vite开发环境

准备 nodejs 18 pnpm https://vitejs.cn/ 开始 pnpm init pnpm add -D vite新建index.html <!DOCTYPE html> <html lang"zh"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width…

飞睿智能无线高速uwb安全数据传输模块,低功耗、抗干扰超宽带uwb芯片传输速度技术新突破

在信息化的时代&#xff0c;数据传输的速度和安全性无疑是每个企业和个人都极为关注的话题。随着科技的飞速发展&#xff0c;超宽带&#xff08;Ultra-Wideband&#xff0c;简称UWB&#xff09;技术凭借其性能和广泛的应用前景&#xff0c;逐渐成为了数据传输领域的新星。今天&…

JavaScript学习笔记(七)

45.9 JavaScript 可迭代对象 可迭代对象&#xff08;Iterables&#xff09;是可以使用 for..of 进行迭代的对象。 从技术上讲&#xff0c;可迭代对象必须实现 Symbol.iterator 方法。 45.9.1 遍历字符串 <body><p id"demo"></p><script>c…

使用vllm部署大语言模型

vLLM是一个快速且易于使用的库&#xff0c;用于LLM&#xff08;大型语言模型&#xff09;推理和服务。通过PagedAttention技术&#xff0c;vLLM可以有效地管理注意力键和值内存&#xff0c;降低内存占用和提高计算效率。vLLM能够将多个传入的请求进行连续批处理&#xff0c;从而…

PTrade常见问题系列5

回测失败&#xff1a;可用资源不足。 回测运行失败&#xff0c;错误码&#xff1a;2 错误信息&#xff1a;可用资源不足&#xff0c;请稍后在创建。 1、之前客户未限制客户容器使用内存和CPU&#xff0c;周末修改配置&#xff0c;限制了内存和CPU&#xff1b; 2、此报错是用户…

【Python】已解决:FileNotFoundError: [Errno 2] No such file or directory: ‘D:\1. PDF’

文章目录 一、分析问题背景二、可能出错的原因三、错误代码示例四、正确代码示例五、注意事项 已解决&#xff1a;FileNotFoundError: [Errno 2] No such file or directory: ‘D:\1. PDF’ 一、分析问题背景 在Python编程中&#xff0c;当你尝试打开一个不存在的文件时&…

索引唯一约束问题SQL

新增报错违反唯一约束条件 (JING_DIAN.SYS_C0096533) 【问题原因】 这个问题可能是由于在Oracle APEX中&#xff0c;虽然你创建了一个名为"ISEQ_520227"的索引&#xff0c;但是在插入数据时&#xff0c;违反了唯一约束条件。这可能是因为你的数据表中已经存在相同的…

压测引擎数据库设计(上)

压测引擎数据库设计&#xff08;上&#xff09; 引言 在当今快速发展的互联网时代&#xff0c;软件质量保证和性能测试变得尤为重要。自动化测试平台&#xff0c;提供了一套完整的解决方案&#xff0c;以确保软件产品在发布前能够满足性能和稳定性的要求。本文将深入探讨滴云自…

jmeter-beanshell学习6-beanshell生成测试报告

前面写了各种准备工作&#xff0c;内容组合用起来&#xff0c;应该能做自动化了&#xff0c;最后一步&#xff0c;生成一个报告&#xff0c;报告格式还是csv 报告生成的路径和文件&#xff0c;在用户参数写好&#xff0c;防止以后改路径或者名字&#xff0c;要去代码里面改。以…

Django自动生成Swagger接口文档 —— Python

1. 前言 当接口开发完成&#xff0c;紧接着需要编写接口文档。传统的接口文档通常都是使用Word或者一些接口文档管理平台进行编写&#xff0c;但此类接口文档维护更新比较麻烦&#xff0c;每次接口有变更&#xff0c;需要手动修改接口文档。在实际的工作中&#xff0c;经常会遇…

tomcat的优化和tomcat和nginx实现动静分离:

tomcat的优化 tomcat自身的优化 tomcat的并发处理能力不强。大项目不使用tomcat做为转发动态的中间件&#xff08;k8s集群&#xff0c;python&#xff0c;rubby&#xff09;&#xff0c;小项目会使用&#xff08;内部使用&#xff09;&#xff0c;动静分离。 优化tomcat的启动…

Python入门 2024/7/8

目录 数据容器 dict(字典&#xff0c;映射) 语法 定义字典字面量 定义字典变量 定义空字典 从字典中基于key获取value 字典的嵌套 字典的常用操作 新增元素 更新元素 删除元素 清空字典 获取全部的key 遍历字典 统计字典内的元素数量 练习 数据容器的通用操作…

在公司的业务杂记1之多选部门且主表没有部门字段(子表查询)

原型 1.新建&#xff0c;上传报告可多选部门 2.查询&#xff0c;可多选部门 数据库&#xff08;Postgresql&#xff09; 方式一 新增字段Jsonb&#xff1a; CREATE TABLE public.admin_report (admin_report_uuid uuid DEFAULT gen_random_uuid() NOT NULL,admin_report_tit…

java —— JSP 技术

一、JSP &#xff08;一&#xff09;前言 1、.jsp 与 .html 一样属于前端内容&#xff0c;创建在 WebContent 之下&#xff1b; 2、嵌套的 java 语句放置在<% %>里面&#xff1b; 3、嵌套 java 语句的三种语法&#xff1a; ① 脚本&#xff1a;<% java 代码 %>…

安全防御第三天(笔记持续更新)

1.接口类型以及作用 接口 --- 物理接口 三层口 --- 可以配置IP地址的接口 二层口 普通二层口 接口对 --- “透明网线” --- 可以将一个或者两个接口配置成为接口对&#xff0c;则 数据从一个接口进&#xff0c;将不需要查看MAC地址表&#xff0c;直接从另一个接口出&#xff1b…

汇川CodeSysPLC教程 Modbus变量编址

线圈&#xff1a;位变量&#xff0c;只有两种状态0和1。汇川PLC中包含Q区及SM区等变量。 寄存器&#xff1a;16位&#xff08;字&#xff09;变量&#xff0c;本PLC中包含M区及SD区等变量 说明&#xff1a; 汇川HMI的专用协议使用不同功能码&#xff1a;在访问SM时&#xff0c…

Python--并发编程--协程

概念 协程是轻量级的线程&#xff0c;它是程序员管理的并发机制&#xff0c;使得在一个线程中程序可以在多个函数之间交替运行。 Python中主要通过asyncio模块实现协程。 协程函数 用async修饰的函数 import asyncio# func为协程函数 async def func():await asyncio.slee…

2024HW必修高危漏洞集合_v4.0

高危风险漏洞一直是企业网络安全防护的薄弱点&#xff0c;也成为HW攻防演练期间红队的重要突破口;每年 HW期间爆发了大量的高危风险漏洞成为红队突破网络边界防护的一把利器,很多企业因为这些高危漏洞而导致整个防御体系被突破、甚至靶标失守而遗憾出局。 HW 攻防演练在即&…

如何做一个透明度渐现且向上位移逐行出现的文字效果

前言 在这个夜黑风高的夜晚&#xff0c;你的眼睛已经开始有些疲惫。你的手指在键盘上轻轻地敲击着&#xff0c;仿佛在弹奏一首无声的夜曲。你的思绪在代码的海洋中飘荡&#xff0c;寻找着最后一行需要完成的代码。就在这时&#xff0c;你的老板走了过来&#xff0c;他的脸上带…