C++11:继承

2023-05-16

目录

继承的基本概念

继承方式

 基类和派生类对象赋值转换/切片

继承中的作用域

派生类的四个成员函数:

 构造函数

拷贝构造函数

赋值重载

析构函数

静态成员

继承与友元

多继承

菱形继承

多继承的指针偏移问题

 组合


继承的基本概念

继承出现的契机是某一些类中会有一部分相似的信息,我们希望以这个类的基本信息为基础能生成更多高分化的类。

集体来讲就是具有一些公共属性的类我们不希望多次去写,比如说人们的职位分布,其主要重合的信息比如名字电话号码性别等等

具体的操作流程如下,我们创建一个叫Person的类,然后使用public的继承方式将其继承到worker这个子类上

当然继承的术语还有其他叫法,不过都一样,喜欢哪个用哪个

继承的使用格式:

 子类会会继承获得父类的成员变量以及成员函数

好,既然牵扯到了访问限定符的问题,那么对于子类,它的访问方式是怎么做的呢?或者说它的访问规则是怎么样的?

继承方式

 

由于访问限定符和继承方式33相乘,有9种继承方式,不过说明白还是比较简单的

基类中的私有其本意是:“我不想给你继承”,我们调试看看

  • 诶?不对啊,不是说不可见吗,这不还是继承下来了吗?这里造成误解的原因则是关于“不可见”的定义问题,虽然private修饰的成员变量是不可见的,但此处的不可见则是指于子类中,无论是类内部还是外部都不能访问这个继承下来的变量。继承是继承下来了,但是它上了层盾,你没法访问。

但是这里还是很奇怪,我为什么还是能借助父类的函数访问到不可见的变量?

 这里的做法确实没有什么问题,但是如果我们想在子类去访问就不行了。

那么有没有比较中立一点的?protect就可以实现比private宽容一点的访问权限,也就是类外部依旧不能访问,但是类内部可以。

我们在protect下创建一个新的成员变量,然后在派生类的成员函数中去访问它。

那么上面的情况都是在public的条件下派生类继承基类时的权限问题,那么当我们更换继承方式的时候是什么样的?

当我们使用Protect的方式访问时,位于public下的函数不能使用了,但是protect内部的依然可以private依旧不行

由于重复内容比较多,就借用一下表格了。

总结起来:

  • 当以public方式继承时,子类可以访问到父类public下的成员函数以及成员变量,也可以访问到portected下的变量和函数,private不行。
  • 当以protected方式继承时,子类不能访问到父类访public下的成员函数以及成员变量,但可以访问到portected下的变量和函数,private不行。
  • 当以protected方式继承时,子类不能访问到父类访public下的成员函数以及成员变量,依然可以访问到portected下的变量和函数,private不行。

需要注意的是:访问限定符可以不写

使用关键字class时默认的继承方式是private,使用struct时默认的继承方式是public,不过最好显示的写出继承方式

在实际运用中一般使用都是public继承,几乎很少使用protetced/private继承,也不提倡使用protetced/private继承,因为protetced/private继承下来的成员都只能在派生类的类里面使用,实际中扩展维护性不强

 总结继承的实用性:继承的这个复用功能如果去细品会发现它还是能节省不少工作流程的,我们仅需要一个主体框架,当我们需要以这个框架为基础生成更加多功能的组件或者说类的时候,复用就会变得方便许多,毕竟我们不需要再回头去改类内部的成员变量函数什么的。

 基类和派生类对象赋值转换/切片

 既然使用了继承,那么子类一般情况下一定会比父类多一点成员变量什么的,在这种情况下假如将子类赋值回给父类生成的对象,父类的引用,父类的指针时,就会产生一个类似切片的过程

 非常形象,子类对象反赋值回父类对象直接将子类对象中多余的部分切去,保留对应的父类成员变量。

注意!在这个过程中没有类型转换!也没有临时对象,根据我们之前学习的内容,两个不同类型的类是不能相互赋值的,能发生类型转换的是单参构造函数虽然可以发生,在这个过程中,单一变量被拿去构造了一个相同类型的临时对象,这个临时对象去赋值给新的对象。

而且仅针对子类向上赋值给父类,不能父类给子类。

所以这个赋值的过程是不会生成临时变量的。

那么看上去也没什么影响嘛!不就是少了个临时对象么?

临时对象其实在很多方面会比较碍事,比如如下,我们想引用一下这个j,但是其中产生的临时变量具有常性,所以不行,加个const才行。

举个例子:

 而父类子类这个过程就不需要考虑了,直接爽用

但是也是有代价的,赋值或者引用发生时,它仅生效于父类子类相同的那一部分

而且继承赋值时,会调用父类的拷贝构造函数

继承中的作用域

 有个小问题,当我们的子类和父类中都有一个同名的变量时,访问哪一个?

在子类访问用子类的,父类访问用父类的

当然想要在子类访问父类的也不是不行,加个作用于限定符即可。

其实发生这种现象的原因是当出现同名对象时,子类会隐藏父类的同名成员,但不是不让访问

那么刚才是同名的成员变量,那么假如现在是同名成员函数会怎么处理?

首先一定会触发隐藏,跟同名的成员变量没什么区别。

但是如果它写成重载的样式呢??刁钻的老6就来了。

那么它们构成重载还是重写呢?还是隐藏(重定义)或者编译报错?

  • 首先,重载的发生条件是处于同一作用域下才会重载。
  • 那么他就不是能重载的,如下这样使用时就是单纯的隐藏。

 以函数调用为举证,当前的Text需要传参

 构成了隐藏,没法调到A的函数,被隐藏了要加作用域限定符才是。

派生类的四个成员函数:

 既然派生类类会继承基类的对象,那么它里面的各个成员函数的工作则是不同的。

 构造函数

  • 不同于我们创建一个新的对象,对于派生类来说,基类的对象派生类只能去调用基类的构造函数才能初始化,同理,假如基类对象其中有需要被清理的资源也只能调用父类的析构函数
  • 派生类的构造函数必须调用基类的构造函数初始化基类的那一部分成员。如果基类没有默认
  • 的构造函数,则必须在派生类构造函数的初始化列表阶段显示调用。

我们就尝试初始化一下基类的对象

  •  但是想要去初始化它是不可以通过直接访问的方式来进行的,要调用基类的构造函数才行。

 调用父类的构造函数

拷贝构造函数

  •  拷贝构造也是同理,基类的对象需要调用基类的拷贝构造来拷贝,派生类的则是处理自己的对象。派生类的拷贝构造函数必须调用基类的拷贝构造完成基类的拷贝初始化
     

 当然,我们也可以借助切片的行为来实现拷贝构造,切片的行为会自己调用父类的拷贝构造函数。

赋值重载

子类的和父类构成隐藏,派生类的operator=必须要调用基类的operator=完成基类的复制

析构函数

已我们前面的理解,既然构造等默认成员函数们都是以各自管各自类的工作方法执行它们的功能,那么析构函数也应该是子类析构自己的,然后调用父类的析构去析构父类成员。

但是析构函数有个让人不解的点:为什么没办法调用父类的构造?

 这样就可以了

 

 为什么加了个作用域就可以了?

  • 因为子类析构函数和父类析构函数构成了隐藏关系(由于后面多态的关系需求,所有的析构函数都会被特殊处理成为相同名字的函数,然后构成隐藏)

但是还有一个及其奇怪的点,为什么我们在指定作用域只调用了一次父类的析构函数,结果调用了两次父类的析构函数?甚至还崩溃了。

继承结构下的析构函数的特点:因为子类的析构函数会自己默认调用父类的析构函数

  • 子类的析构函数和父类的析构函数默认情况下会构成隐藏关系,从调用角度来讲,我们没法直接调用到基类的析构函数来析构基类的对象。(回顾一下:函数名相同时构成隐藏)但这根本名字不一样啊?原因就是为了多态而服务的,因为后续一些场景析构函数需要构成重写,重写的条件之一是函数名相同。那么编译器会对析构函数名进行特殊处理,处理成destrutor(),所以父类析构函数不加virtual的情况下,子类析构函数和父类析构函数构成隐藏关系,这个问题要先遗留上一阵子了。

所以总结一下,子类的默认成员函数,构造和赋值都要显示调用,而析构则不用


做个小题目巩固以下知识吧!

下面说法正确的是( )

A.派生类构造函数初始化列表的位置必须显式调用基类的构造函数,已完成基类部分成员的初始化

B.派生类构造函数先初始化子类成员,再初始化基类成员

C.派生类析构函数不会自动析构基类部分成员

D.子类构造函数的定义有时需要参考基类构造函数

 


答案以及解析:

A.如果父类有默认构造函数,此时就不需要

B.顺序相反,先初始化父类,再是子类

C.会调用,并且按照构造的相反顺序进行调用

D.是的,需要看父类构造函数是否需要参数子类的,从而你决定子类构造函数的定义

静态成员

  • 当继承发生时,其静态成员还是同一个,而非静态成员则是各自的。
  • 究其原因,静态成员是属于整个类中所有的对象,同时也属于所有的派生类。

继承与友元

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

多继承

在继承的方法后面加上一个逗号就可以实现多继承

 

 比如说一个市民,它可以复用一个人类,然后再复用一个工人类,这其实还不错,但是非常阴间的玩意马上就来了,菱形继承

菱形继承

 

 

菱形继承会造成数据冗余以及二义性

  • 以二义性为例:以上的这个例子,相当于不仅继承了strudent的Person类,又继承了Teacher的Person类,假如我们直接去访问基类中的对象时,会直接报错,因为编译器也不知道你想访问那个Person类里头的对象。

我们构建一个简单的菱形继承,然后在内存中观察一下它真实的空间分配

 

 

  • 我们在内存之中看到了菱形继承的大致模型,这个模型本身并不复杂,而且非常直观,BC两个对象之中都各有一个a成员,他们是各自独立开来的。
  • 那么当我不想要独立开这个a的时候该怎么办呢?接下来的情况就有些复杂,我们对中介类加上一个virtual

 

  • 我们发现,a此时没有独立开来,而是算作了整个菱形继承的公共对象,无论在哪更改a,都指向了单独的一个a,而非独立开来的a。
  • 那么问题来了,原本的a的地址变成了什么?

 我们发现在内存中他们的字节数都是一样的,这个其实就是距离虚基类对象的偏移量

 这种记载偏移量的方法可以直接让我们使用菱形继承的时候编译器可以非常精确的寻找到当前基对象的位置并更改以解决二义性的问题。

多继承的指针偏移问题

多继承时,一个继承了多个类的对象在使用其中一个父类指针产生切片时,可能会发生指针偏移

借助一道题可以很好的理解这个问题

 那么P1 P2 的指向模型如下

 

 组合

 什么是组合?

 以上图为例,就是在一个类里面已某个自定义类型再创建了一个对象

  • 他和继承都是可以实现复用的,但是相较于继承,组合不能使用父类被保护的成员,
  • 那么有人提出了一个概念,继承称之为:白箱复用,而组合则叫做:黑箱复用
  • 黑和白的区别主要还是区别于能不能看见其中一些内部的使用方法。
  • 平常来讲,组合的效果会更加好,因为黑箱复用的耦合度较低。
  • 耦合度指的是关联互相影响度,牵连的东西的多少。
  • 我们以继承为例,由于是白箱复用,当我们更改父类内部的成员时,其子类绝对会受到影响,而对于组合来说则不会,毕竟没有对你门洞大开,其耦合度低,基本上影响不到组合的对象。

 

以上就是C++继承中部分的知识了

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

C++11:继承 的相关文章

  • 加速度计、陀螺仪工作原理

    加速度计 陀螺仪的工作原理 参考链接 xff1a https c miaowlabs com B07 html 陀螺仪 加速度计都是惯性测量元件的一种 而 MPU 6050 传感器的内部同时集成了陀螺仪和加速度传感器两种惯性测量元件 1 加
  • VsCode中运行C/C++

    VsCode中运行C C 43 43 1 插件 runCode2 配置环境 mingw64 1 插件 runCode 在 VsCode 中的扩展商店中 xff0c 下载插件 Code Runner 安装完成之后 xff0c 进行一些配置更改
  • 常见通信协议之UART、RS485

    UART 通用异步收发器一种通用的串行 异步通信总线 xff0c 该总线有两条数据线 xff0c 可以实现全双工的发送和接受并行通信和串行通信 总线传递数据的本质 高低电信号并行通信 一次性传输多个位 布线难度高 存在数据干扰串行通信 逐次
  • java的琐碎学习之串口通信与数据库与GUI

    RFID作业 xff0c 要求实现软硬结合 xff0c 全部使用自己的页面完成 xff1b 找了几个教程发现安卓我做不到 xff0c 就用了Java实现 xff1b 图书管理系统 可以通过写卡来绑定15693卡和书籍 xff0c 实现增删改
  • C++- #define 和 const 有什么区别?

    回答如下 xff1a 定义不同 xff1a define 是C 43 43 预处理器的指令 xff0c 用于定义宏 xff0c const是C 43 43 关键字 xff0c 用于定义常量 作用对象不同 xff1a define 定义的宏
  • HTTP协议:二.使用工具观察 HTTP 的请求和响应

    二 使用工具观察 HTTP 的请求和响应 1 HTTP 协议格式 HTTP 是一个文本格式的协议 可以通过 Chrome 开发者工具或者 Fiddler 抓包 分析HTTP 请求 响应的细节 2 抓包工具的下载和使用 直接去官网下载即可 f
  • Linux环境下的c语言编程

    vim编辑器编辑hello c vim编辑器中输入相应代码 编译 运行代码 运行结果 使用GDB函数调用 编译生成可执行文件 启动gdb 第十行设置断点并运行 gcc过程改为makefile管理 编写makefile文件 启动makefil
  • ubuntu下关于ssh远程和scp远程复制文件以及nfs搭建

    SSH远程 在Linux系统中 xff0c 通过客户端连接到远程服务器中 xff0c 方便代码地编写运行 xff0c ssh是一种安全协议 xff0c 主要用于给远程登录信息数据进行加密 1 安装ssh 2 启动ssh 3 创建要发送的文件
  • Linux环境下的多线程&多进程编程

    1 线程的创建与终止 创建一个 c文件 xff0c 使用vi编辑器进行多线程的创建 编译文件 在编译文件时会出现对 pthread create 未定义的引用 xff0c 这是由于pthread 库不是Linux系统默认的库 xff0c 连
  • 东北天坐标系转载体坐标系

    文章目录 1 基本概念1 1欧拉角1 2左乘右乘1 3东北天坐标系1 4载体坐标系1 5捷联惯性导航系统 2 通过ECEF转换到参考点附近的ENU坐标系上3 东北天坐标系到载体坐标系 1 基本概念 1 1欧拉角 欧拉旋转定理指出 xff1a
  • I2C驱动App

    1 查看eeprog c源代码 copyright C by 2009 Guangzhou FriendlyaRM in China email capbily 64 163 com website arm9 net include lt
  • Qt5.14.2 编程应用

    Qt5 14 2 编程应用 什么是Qt Qt 是一个跨平台的 C 43 43 图形用户界面库 xff0c 由挪威 TrollTech 公司于 1995 年底出品 xff0c 并于 2008年6月17日被NOKIA公司收购 xff0c 以增强
  • L298N电机驱动的使用

    L298N电机驱动的使用 前言一 介绍L298N模块简介接口介绍 二 使用步骤硬件连接软件部分1 声明部分2 代码部分 总结 前言 博主为某大学电气专业大学生 xff0c 以学习为目的写下该文 xff0c 内容主要为以51单片机为例简单介绍
  • Authorization头的作用

    Authorization头的主要用作http协议的认证 Authorization的作用是当客户端访问受口令保护时 xff0c 服务器端会发送401状态码和WWW Authenticate响应头 xff0c 要求客户机使用Authoriz
  • vscode中常用的快捷键

    分享一些本人在学习前端过程中用到的一些快捷键 xff0c 需要强调的是 xff0c 这些快捷键适用的软件是VScode 因为自己初学前端用的是这个软件 其中有一些在idea中也是适用的 xff0c 已经在括号内标注 1 alt 43 W 将
  • PID算法原理及基本实现

    自动控制中 xff0c PID及其衍生出来的算法是应用最广的算法之一 各个做自动控制的厂家基本都有会实现这一经典算法 我们在做项目的过程中 xff0c 也时常会遇到类似的需求 xff0c 所以就想实现这一算法以适用于更多的应用场景 1 PI
  • Spring Boot基础学习之(六):前后端交互实现用户登录界面

    本次项目所有能够使用的静态资源可以免费进行下载 静态资源 本篇博客写的内容 xff0c 是一个系列 xff0c 内容都是关于spring boot架构的学习 xff0c 实现前后端交互 xff0c 极大的解放双手spring boot学习系
  • USMART调试组件

    什么是USMART USMART是正点原子团队为其STM32开发平台开发的一种类似linux的shell的调试工具 具体工作过程是通过串口发送命令给单片机 然后单片机收到命令之后调用单片机里面对应的相关函数 并执行 同时支持返回结果 USM
  • 内部温度传感器

    STM32有一个内部的温度传感器 可以用来测量CPU及周围的温度 TA 该温度传感器在内部和ADCX IN16输入通道相连接 此通道把传感器输出的电压转换成数字值 温度传感器模拟输入推荐采样时间是17 1us STM32的内部温度传感器支持
  • 光敏传感器

    光敏传感器是利用光敏元件将光信号转换为电信号的传感器 它的敏感波长在可见光波长附近 包括红外线波长和紫外线波长 光传感器不只局限于对光的探测 它还可以作为探测元件组成其他传感器 对许多非电量进行检测 只要将这些非电量转换为光信号的变化即可

随机推荐