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

【C++技能树】继承概念与解析

在这里插入图片描述
Halo,这里是Ppeua。平时主要更新C++,数据结构算法,Linux与ROS…感兴趣就关注我bua!

继承

  • 0. 继承概念
    • 0.1 继承访问限定符
  • 1. 基类和派生类对象赋值兼容转换
  • 2. 继承中的作用域
  • 3. 派生类中的默认成员函数
  • 4.友元
  • 5.继承中的静态成员
  • 6.菱形继承
  • 7.菱形虚拟继承
  • 总结
  • 总结

在这里插入图片描述

0. 继承概念

​ 设想一个场景,你需要设计学生、老师、教授…的类,除了每个身份中独有的信息,例如:学号,工号,教授身份号,但是他们都有一个共同的属性,就是人.所以我们可以先设计一个类:人.

每设计一个新的类都可以复用人这个类,增加了代码的复用性.这就是C++中的新特性:继承

我们之前接触的函数重载是函数层面的复用,继承则是类层面的复用

class Person{
public:void print(){cout<<age<<" "<<name<<endl;}int age=0;string name="Peter";string address;int tel;
};
class Student:public Person
{
public:int _stuid;
};

我们可以通过调用来看看其结构模型.

int main()
{Person p1;cout<<p1.name<<endl;Student s1;cout << s1._stuid;
}

image-20230823184212791

子类中可以共享父类中的变量,父类不可以访问子类的变量.

0.1 继承访问限定符

在类中有访问限定符,同样的,在继承方式上也有访问限定符.

24756bb84f210a146239f27a1e1494b

派生类(子类)可以通过以下方式来继承基类(父类):

image-20230823193458458

在这里新出现了一个权限符号,protected.它与private是类似的:

protected修饰的变量在类外与private类似,不能被访问.

但是在派生类中可以访问protected修饰的变量,而不能访问private修饰的变量

所以权限的大小的关系为:

public>protected>private

所以,在权限的继承中有一个最小原则.

  1. 以public来继承,可以继承的变量为:public,protected
  2. 以protected来继承.可以继承的变量为:protected
  3. 以private来继承,无可以继承的变量

通常情况下,我们一般用public来继承,protected/private的继承方式实用性不高

1. 基类和派生类对象赋值兼容转换

派生类可以转换为基类,而基类并不能转换为派生类.

例如上面Person与Student的例子:

可以实现
Person p1;
Student s1;
p1=s1;
不能实现
s1=p1;

这其实很好理解.父类中的属性往往比子类中的成员多,子类中的成员可以通过切割多余的成员转换到父类中.

通过这样的方式,这中间不涉及强制类型转换.我们可以通过以下这个例子来看.

我们知道,强制类型转换会产生一个临时变量.例如:

int a=10;
double b=a;

这当中会产生一个临时变量double a,来赋值给b.

Person &p1=s1;

如果产生了临时变量,这个赋值是不可以的.但通过编译器验证,我们发现这样是可以的.侧面的说明这并不是引用.

但对于p1的成员进行修改,s1也会同样被修改.

Student s1;
Person p1 = s1;
p1.name = "H";

Before:

image-20230823231534645

After:

image-20230823231912045

虽然这种限制(子类可以转父类,父类不可转子类)可以通过指针直接访问内存的方法解除

原来指向的是student对象,现在强制转换为student指针是可以的.

Student s1;
Person *p1 = &s1;
Student* sp1 = (Student*)p1;
sp1->age = 10;

原来指向的是person对象,现在强制转换为student指针则会发生越界

Student s1;
Person pp1;
Person *pp1 = &s1;
Student* sp1 = (Student*)pp1;
sp1->age = 10;

2. 继承中的作用域

一个{}是一个作用域,所以在基类和派生类中,都有自己的作用域.

所以当在派生类中定义与基类相同的名的变量的时候就会构成隐藏:隐藏父类的相关变量

当在派生类中定义与基类相同的函数时(只需要名字相同),就会构成重定义(隐藏):重定义父类相关函数

如果需要访问父类被隐藏的属性,需要在前加类域限定符才能访问

#include<iostream>
using namespace std;
class Person {
public:void print(){cout <<"Person:" << age << endl;}int age = 0;
};
class Student :public Person
{
public:void print(int i){cout <<"Student:" << age << endl;}int age=10;};
int main()
{Student s1;cout << s1.age; //10s1.print(1);// student:10cout << s1.Person::age;//0s1.Person::print();//person:0return 0;
}

3. 派生类中的默认成员函数

总的来说,派生类中的所有涉及父类的行为都要从父类当中去寻找相关方法论:

  1. 派生类初始化的时候会先调用父类的初始化函数,在调用自己的.若父类没有默认构造函数则需要在派生类中的初始化列表中调用父类构造函数传入参数.(**为什么需要在初始化列表中调用不在函数体里调用呢?**自定义类型成员(且该类没有默认构造函数时)在初始化列表中调用相关文章:初始化列表

    class Person{
    public:Person(int sage):age(sage){}void print(){cout<<age<<" "<<name<<endl;}int age=0;string name="Peter";string address;int tel;
    private:int s=0;
    };
    class Student:public Person
    {
    public:Student(int _age):Person(_age){}int _stuid;
    };
    
  2. 拷贝构造与赋值运算符重载需要通过显式调用父类中的方法来完成.

    class person
    {
    public:person(const char *name="peter"):_name(name){cout<<"person()";}person(const person&p1):_name(p1._name){cout<<"person(const Person& P)"<<endl;}person& operator=(const person&p){cout<<"person operator=(const person&p)"<<endl;if(this!=&p)_name=p._name;return *this;}string _name;};
    class student:public person
    {
    public:student(const char*name="zhangsan",int id=0):person(name),_id(id){cout<<"student()"<<endl;}student(const student&s1):person(s1),_id(s1._id){}student& operator=(const student& s1){if(this!=&s1){//出现隐藏,想要调用父类的=person::operator=(s1);_id=s1._id;}return *this;}void print(){cout<<_id<<" "<<_name<<endl;}
    private:int _id;
    };
    
  3. 析构函数不需要显式调用父类(也不能),编译器会自己调用完派生类的析构函数,在调用基类的析构函数

    其实这也很好理解.从函数栈帧方面:先创建父类再创建子类,自然先析构子类再析构父类.

    从内存保护方面:在子类中有可能调用了父类的成员对象,如果先消除父类,会导致子类中出现野指针的情况

16375783a643a74a0431af96eefa36c

4.友元

父类的友元不能访问子类的成员变量。(父亲的朋友不是孩子的朋友

class B;
class A{
friend void print(const A& a1,const B& b1);
private:int a=10;};
class B:public A{
private:int b=100;
};
void print(const A& a1,const B&b1)
{cout<<a1.a<<endl; //rightcout<<b1.a<<endl; //rightcout<<b1.b<<endl; //error
}

在上面的例子中可以看到:print函数可以访问A的private,而不能访问B中的private

5.继承中的静态成员

静态成员只会存在一份.在父类当中,子类中可以继承静态成员.但是继承的是访问权,只能访问不能修改

且其是存在类当中,也就是无论几个对象,访问的都是同一个静态成员

class B;
class A{
friend void print(const A& a1,const B& b1);
public:static int count;
private:int a=10;
};
int A::count=10;
class B:public A{
public:void print(){cout<<count<<" "<<endl;}private:int b=100;};void print(const A& a1,const B&b1)
{cout<<a1.count<<endl;cout<<b1.count<<endl;cout<<A::count;
}int main()
{A a1;B b1;print(a1, b1);//10A::count++;print(a1, b1);//11}

6.菱形继承

在c++中,多继承的结构模型是这样的,使用不当时会导致出现菱形继承的情况.导致内存中会重复出现一些变量.也会导致二义性

cdd3d5768cf48df625009ef25537898

例如,在person中有一个表示年龄的age,在student与teacher中各有表示年龄的age,当professor继承student与teacher时,就会有两个age.这在现实环境中显然是不合理的

image-20230824171105761

#include<iostream>
using namespace std;
class Person {
public:int age = 10;
};class Student :public Person
{
public:int stuid = 1;};
class Teacher :public Person
{int teaid = 2;
};
class Professor:public Student,public Teacher
{int profeid = 3;
};
int main()
{Professor p1;p1.Student::age = 100;p1.Teacher::age = 200;}

其在内存中的模型为:

94c65b2a5cf3f4035fcc7ea7cd1efe6

可以看到此时出现数据冗余二义性.

C++解决这个问题的方法则是:菱形虚拟继承

7.菱形虚拟继承

一个新的关键字:virtual,在之后用到很多,但每个地方的含义都不大相同.

在继承方面,我们用virtual来修饰基类.也就是在上方结构模型的腰部

class Person {
public:int age = 10;
};class Student :virtual public Person
{
public:int stuid = 1;};
class Teacher :virtual public Person
{int teaid = 2;
};
class Professor:public Student,public Teacher
{int profeid = 3;
};
int main()
{Professor p1;p1.Student::age = 100;p1.Teacher::age = 200;cout<<sizeof(p1);
}

此时的内存模型为

b991e55de33001344c16e25553e8e45

结构模型为:

image-20230824174055432

我们发现,重复出现的变量age修饰完只出现了一个.

观察内存模型,我们发现原来存age的地方,变成了一个指针.而age被放在了整个对象的最后一个位置.

f65d42e29d4e453b35e1d5ff4da514c

打开内存模型,我们发现,这个指针指向了一块内存空间.叫虚基表.其中第一个位置为:0(其存放的为虚表偏移量) 第二个位置存着该指针相较于age的偏移量

为什么要这样设计呢:

  1. 解决了数据冗余

  2. 相同的对象可以调用同一个虚基表

  3. 存放偏移量让切割成为了可能

    这里可以这样理解,当我创建了一个teacher的对象,将professor传入,则完成了切割,数据从teacher的指针开始访问,若我此时想要访问a,我直接读取偏移量即可.

总结

继承使C++底层变得复杂了起来,在日常使用中,需要避免出现菱形继承的问题.

更推荐使用组合:在一个类中调用另一个封装完的类,此时被调用的类的细节对调用类来说是不可见的.

相较于继承,更推荐使用组合的方式.高内聚低耦合一直是我们设计程序的原则

型,我们发现原来存age的地方,变成了一个指针.而age被放在了整个对象的最后一个位置.

[外链图片转存中…(img-WM0k8lWU-1692871122130)]

打开内存模型,我们发现,这个指针指向了一块内存空间.叫虚基表.其中第一个位置为:0(其存放的为虚表偏移量) 第二个位置存着该指针相较于age的偏移量

为什么要这样设计呢:

  1. 解决了数据冗余

  2. 相同的对象可以调用同一个虚基表

  3. 存放偏移量让切割成为了可能

    这里可以这样理解,当我创建了一个teacher的对象,将professor传入,则完成了切割,数据从teacher的指针开始访问,若我此时想要访问a,我直接读取偏移量即可.

总结

继承使C++底层变得复杂了起来,在日常使用中,需要避免出现菱形继承的问题.

更推荐使用组合:在一个类中调用另一个封装完的类,此时被调用的类的细节对调用类来说是不可见的.

相较于继承,更推荐使用组合的方式.高内聚低耦合一直是我们设计程序的原则

相关文章:优先使用对象组合,而不是类继承

相关文章:

【C++技能树】继承概念与解析

Halo&#xff0c;这里是Ppeua。平时主要更新C&#xff0c;数据结构算法&#xff0c;Linux与ROS…感兴趣就关注我bua&#xff01; 继承 0. 继承概念0.1 继承访问限定符 1. 基类和派生类对象赋值兼容转换2. 继承中的作用域3. 派生类中的默认成员函数4.友元5.继承中的静态成员6.菱…...

计算机网络 第二节

目录 一&#xff0c;计算机网络的分类 1.按照覆盖范围分 2.按照所属用途分 二&#xff0c;计算机网络逻辑组成部分 1.核心部分 &#xff08;通信子网&#xff09; 1.1电路交换 1.2 分组交换 两种方式的特点 重点 2.边缘部分 &#xff08;资源子网&#xff09; 进程通信的方…...

无涯教程-机器学习 - 矩阵图函数

相关性是有关两个变量之间变化的指示&#xff0c;在前面的章节中&#xff0c;无涯教程讨论了Pearson的相关系数以及相关的重要性&#xff0c;可以绘制相关矩阵以显示哪个变量相对于另一个变量具有较高或较低的相关性。 在以下示例中&#xff0c;Python脚本将为Pima印度糖尿病数…...

Redis 高可用与集群

Redis 高可用与集群 虽然 Redis 可以实现单机的数据持久化&#xff0c;但无论是 RDB 也好或者 AOF 也好&#xff0c;都解决 不了单点宕机问题&#xff0c;即一旦单台 redis 服务器本身出现系统故障、硬件故障等问题后&#xff0c; 就会直接造成数据的丢失&#xff0c;因此需要…...

修改文件名后Git仓上面并没有修改

场景&#xff1a; 我在本地将文件夹名称由Group → group ,执行git push 后&#xff0c;远程分支上的文件名称并没有修改。 原因&#xff1a; 是我绕过了git 直接使用了系统的重命名操作。 在 Git 中&#xff0c;对于已经存在的文件或文件夹进行大小写重命名是一个敏感的操作…...

Linux 信号

目录 基本概念信号的分类可靠信号与不可靠信号实时信号与非实时信号 常见信号与默认行为进程对信号的处理signal()函数sigaction()函数 向进程发送信号kill()函数raise() alarm()和pause()函数alarm()函数pause()函数 信号集初始化信号集测试信号是否在信号集中 获取信号的描述…...

深入探讨梯度下降:优化机器学习的关键步骤(二)

文章目录 &#x1f340;引言&#x1f340;eta参数的调节&#x1f340;sklearn中的梯度下降 &#x1f340;引言 承接上篇&#xff0c;这篇主要有两个重点&#xff0c;一个是eta参数的调解&#xff1b;一个是在sklearn中实现梯度下降 在梯度下降算法中&#xff0c;学习率&#xf…...

高频算法面试题

合并两个有序数组 const merge (nums1, nums2) > {let p1 0;let p2 0;const result [];let cur;while (p1 < nums1.length || p2 < nums2.length) {if (p1 nums1.length) {cur nums2[p2];} else if (p2 nums2.length) {cur nums1[p1];} else if (nums1[p1] &…...

Hive-启动与操作(2)

&#x1f947;&#x1f947;【大数据学习记录篇】-持续更新中~&#x1f947;&#x1f947; 个人主页&#xff1a;beixi 本文章收录于专栏&#xff08;点击传送&#xff09;&#xff1a;【大数据学习】 &#x1f493;&#x1f493;持续更新中&#xff0c;感谢各位前辈朋友们支持…...

css transition 指南

css transition 指南 在本文中&#xff0c;我们将深入了解 CSS transition&#xff0c;以及如何使用它们来创建丰富、精美的动画。 基本原理 我们创建动画时通常需要一些动画相关的 CSS。 下面是一个按钮在悬停时移动但没有动画的示例&#xff1a; <button class"…...

LeetCode 面试题 02.05. 链表求和

文章目录 一、题目二、C# 题解 一、题目 给定两个用链表表示的整数&#xff0c;每个节点包含一个数位。 这些数位是反向存放的&#xff0c;也就是个位排在链表首部。 编写函数对这两个整数求和&#xff0c;并用链表形式返回结果。 点击此处跳转题目。 示例&#xff1a; 输入&a…...

一米脸书营销软件

功能优势 JOIN ADVANTAGE HOME PAGE MARKETING 公共主页营销 可同时对多个账户公共主页评论&#xff0c;点赞等 可批量邀请多个好友对Facebook公共主页进行评论点赞等&#xff0c;也可批量登录小号对自己公共主页进行点赞。 GROUP MARKETING 小组营销 可批量针对不同账户进行…...

vue 根据数值判断颜色

1.首先style样式给两种颜色 用:class 三元运算符判断出一种颜色 第一步&#xff1a;在style里边设置两种颜色 .green{color: green; } .orange{color: orangered; }在取数据的标签 里边 判断一种颜色 :class"item.quote.current >0 ?orange: green"<van-gri…...

Hugging Face 实战系列 总目录

PyTorch 深度学习 开发环境搭建 全教程 Transformer:《Attention is all you need》 Hugging Face简介 1、Hugging Face实战-系列教程1&#xff1a;Tokenizer分词器&#xff08;Transformer工具包/自然语言处理&#xff09; Hungging Face实战-系列教程1&#xff1a;Tokenize…...

国标视频云服务EasyGBS国标视频平台迁移服务器后无法启动的问题解决方法

国标视频云服务EasyGBS支持设备/平台通过国标GB28181协议注册接入&#xff0c;并能实现视频的实时监控直播、录像、检索与回看、语音对讲、云存储、告警、平台级联等功能。平台部署简单、可拓展性强&#xff0c;支持将接入的视频流进行全终端、全平台分发&#xff0c;分发的视频…...

HTML <th> 标签

实例 普通的 HTML 表格,包含两行两列: <table border="1"><tr><th>Company</th><th>Address</th></tr><tr><td>Apple, Inc.</td><td>1 Infinite Loop Cupertino, CA 95014</td></tr…...

HTTP/1.1协议中的响应报文

2023年8月30日&#xff0c;周三下午 目录 概述响应报文示例详述 概述 HTTP/1.1协议的响应报文由以下几个部分组成&#xff1a; 状态行&#xff08;Status Line&#xff09;响应头部&#xff08;Response Headers&#xff09;空行&#xff08;Blank Line&#xff09;响应体&a…...

TDengine函数大全-选择函数

以下内容来自 TDengine 官方文档 及 GitHub 内容 。 以下所有示例基于 TDengine 3.1.0.3 TDengine函数大全 1.数学函数 2.字符串函数 3.转换函数 4.时间和日期函数 5.聚合函数 6.选择函数 7.时序数据库特有函数 8.系统函数 选择函数 TDengine函数大全BOTTOMFIRSTINTERPLASTLAS…...

非关系型数据库Redis的安装

一、关系型数据库与非关系型数据库的区别&#xff1a;---------面试高频率问题 1、首先了解一下 什么是关系型数据库&#xff1f; 关系型数据库最典型的数据结构是表&#xff0c;由二维表及其之间的联系所组成的一个数据组织。 优点&#xff1a; 易于维护&#xff1a;都是使用…...

oracle 创建数据库

查询表空间的命令 select t1.name,t2.name from v$tablespace t1,v$datafile t2 where t1.ts# t2.ts#; CREATE TABLESPACE ORM_342_BETA DATAFILE /app/oracle/oradata/sysware/ORM_342_BETA.DBF size 800M --存储地址 初始大小800M autoextend on nex…...

变量 varablie 声明- Rust 变量 let mut 声明与 C/C++ 变量声明对比分析

一、变量声明设计&#xff1a;let 与 mut 的哲学解析 Rust 采用 let 声明变量并通过 mut 显式标记可变性&#xff0c;这种设计体现了语言的核心哲学。以下是深度解析&#xff1a; 1.1 设计理念剖析 安全优先原则&#xff1a;默认不可变强制开发者明确声明意图 let x 5; …...

conda相比python好处

Conda 作为 Python 的环境和包管理工具&#xff0c;相比原生 Python 生态&#xff08;如 pip 虚拟环境&#xff09;有许多独特优势&#xff0c;尤其在多项目管理、依赖处理和跨平台兼容性等方面表现更优。以下是 Conda 的核心好处&#xff1a; 一、一站式环境管理&#xff1a…...

React 第五十五节 Router 中 useAsyncError的使用详解

前言 useAsyncError 是 React Router v6.4 引入的一个钩子&#xff0c;用于处理异步操作&#xff08;如数据加载&#xff09;中的错误。下面我将详细解释其用途并提供代码示例。 一、useAsyncError 用途 处理异步错误&#xff1a;捕获在 loader 或 action 中发生的异步错误替…...

LeetCode - 394. 字符串解码

题目 394. 字符串解码 - 力扣&#xff08;LeetCode&#xff09; 思路 使用两个栈&#xff1a;一个存储重复次数&#xff0c;一个存储字符串 遍历输入字符串&#xff1a; 数字处理&#xff1a;遇到数字时&#xff0c;累积计算重复次数左括号处理&#xff1a;保存当前状态&a…...

鸿蒙中用HarmonyOS SDK应用服务 HarmonyOS5开发一个医院挂号小程序

一、开发准备 ​​环境搭建​​&#xff1a; 安装DevEco Studio 3.0或更高版本配置HarmonyOS SDK申请开发者账号 ​​项目创建​​&#xff1a; File > New > Create Project > Application (选择"Empty Ability") 二、核心功能实现 1. 医院科室展示 /…...

定时器任务——若依源码分析

分析util包下面的工具类schedule utils&#xff1a; ScheduleUtils 是若依中用于与 Quartz 框架交互的工具类&#xff0c;封装了定时任务的 创建、更新、暂停、删除等核心逻辑。 createScheduleJob createScheduleJob 用于将任务注册到 Quartz&#xff0c;先构建任务的 JobD…...

跨链模式:多链互操作架构与性能扩展方案

跨链模式&#xff1a;多链互操作架构与性能扩展方案 ——构建下一代区块链互联网的技术基石 一、跨链架构的核心范式演进 1. 分层协议栈&#xff1a;模块化解耦设计 现代跨链系统采用分层协议栈实现灵活扩展&#xff08;H2Cross架构&#xff09;&#xff1a; 适配层&#xf…...

AI编程--插件对比分析:CodeRider、GitHub Copilot及其他

AI编程插件对比分析&#xff1a;CodeRider、GitHub Copilot及其他 随着人工智能技术的快速发展&#xff0c;AI编程插件已成为提升开发者生产力的重要工具。CodeRider和GitHub Copilot作为市场上的领先者&#xff0c;分别以其独特的特性和生态系统吸引了大量开发者。本文将从功…...

云原生玩法三问:构建自定义开发环境

云原生玩法三问&#xff1a;构建自定义开发环境 引言 临时运维一个古董项目&#xff0c;无文档&#xff0c;无环境&#xff0c;无交接人&#xff0c;俗称三无。 运行设备的环境老&#xff0c;本地环境版本高&#xff0c;ssh不过去。正好最近对 腾讯出品的云原生 cnb 感兴趣&…...

A2A JS SDK 完整教程:快速入门指南

目录 什么是 A2A JS SDK?A2A JS 安装与设置A2A JS 核心概念创建你的第一个 A2A JS 代理A2A JS 服务端开发A2A JS 客户端使用A2A JS 高级特性A2A JS 最佳实践A2A JS 故障排除 什么是 A2A JS SDK? A2A JS SDK 是一个专为 JavaScript/TypeScript 开发者设计的强大库&#xff…...