【c++】继承学习(一):继承机制与基类派生类转换

Alt

🔥个人主页Quitecoder

🔥专栏c++笔记仓

Alt

朋友们大家好,本篇文章我们来学习继承部分

目录

  • `1.继承的概念和定义`
    • `继承的定义`
    • `继承基类成员的访问方式变化`
  • `2.基类和派生类对象赋值转换`
  • `3.继承中的作用域`

1.继承的概念和定义

继承(inheritance)机制是面向对象程序设计使代码可以复用的最重要的手段,它允许程序员在保持原有类特性的基础上进行扩展,增加功能,这样产生新的类,称派生类。继承呈现了面向对象程序设计的层次结构,体现了由简单到复杂的认知过程。以前我们接触的复用都是函数复用,继承是类设计层次的复用

通过继承,子类可以重用父类的代码,这有助于减少代码冗余和复杂性,并增加代码的可复用性

子类和父类是继承关系中的两个基本概念:

  1. 父类/ 基类:
    父类是一个更一般的类,它定义了一种通用的数据类型和方法,这些可以被其他类继承。它是继承关系中处于较高层次的类,其特性(属性和方法)可以传递到派生的类中。其他从父类继承的类会自动获得父类定义的所有公共和受保护的成员。

  2. 子类/ 派生类:
    子类是从一个或多个父类继承特性的类。它是继承关系中处于较低层次的类,可以继承其一或多个父类的属性和方法。子类通常会添加一些特有的属性和方法,或者重写某些从父类继承的方法来改变行为。子类集成了父类的特征,并可以拥有自己的特征。

简单来说,父类是派生过程的起点,提供了基础的属性和方法,而子类是继承的结果,它可以扩展和定制继承来的属性和方法。通过这种方式,子类和父类形成了一种层次结构,允许更高层次的代码重用和泛化

例如下面的例子:

在这里插入图片描述

父类包含一些通用的属性,人名和年龄,派生类继承自父类但具有不同的额外特性或方法

class Person
{
public:void Print(){cout << "name:" << _name << endl;cout << "age:" << _age << endl;}
protected:string _name = "jason"; // 姓名int _age = 18;  // 年龄
};
class Student : public Person
{
protected:int _stuid; // 学号
};class Teacher : public Person
{
protected:int _jobid; // 工号
};

继承后父类的Person的成员(成员函数+成员变量)都会变成子类的一部分。这里体现出了Student和Teacher复用了Person的成员

下面我们使用监视窗口查看Student和Teacher对象,可以看到变量的复用。调用Print可以看到成员函数的复用

int main()
{Student s;Teacher t;s.Print();t.Print();return 0;
}

在这里插入图片描述
在这里插入图片描述

继承的定义

格式

在这里插入图片描述
继承关系和访问限定符:

在这里插入图片描述

继承基类成员的访问方式变化

类成员/继承方式public继承protected继承private继承
基类的public成员派生类的public成员派生类的protected成员派生类的private成员
基类的protected成员派生类的protected成员派生类的protected成员派生类的private成员
基类的private成员在派生类中不可见在派生类中不可见在派生类中不可见
  1. 基类private成员在派生类中无论以什么方式继承都是不可见的。这里的不可见是指基类的私有成员还是被继承到了派生类对象中,但是语法上限制派生类对象不管在类里面还是类外面都不能去访问它

我们前面知道,类里面可以访问它的成员,但是private继承下,子类是无法访问父类的成员的

class Person
{
public:void Print(){cout << "name:" << _name << endl;cout << "age:" << _age << endl;}
protected:string _name = "jason"; // 姓名
private:int _age = 18;  // 年龄
};

我们这个类,拥有三个成员

class Student : public Person
{Student(){_name = "peter";}
protected:int _stuid; // 学号
};

在我们这个子类中,我们可以访问除了父类私有成员的其他成员父类的私有成员父类自己可以用,子类不可以直接使用

但是可以间接使用,比如我用子类来调用上面的Print函数

class Student : public Person
{void Fun(){_name = "abc";Print();}
protected:int _stuid; // 学号
};

在这里插入图片描述

  1. 基类private成员在派生类中是不能被访问,如果基类成员不想在类外直接被访问,但需要在派生类中能访问,就定义为protected可以看出保护成员限定符是因继承才出现的

  2. 实际上面的表格我们进行一下总结会发现,基类的私有成员在子类都是不可见。基类的其他成员在子类的访问方式 == 权限小的那个(成员在基类的访问限定符,继承方式),public > protected > private。

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

class Student : protected Person
{
public:void Fun(){_name = "abc";Print();}
protected:int _stuid; // 学号
};

公有的Print函数遇到protected继承变成保护类,无法外部直接调用:

在这里插入图片描述
保护是类外面不能访问,类里面还可以访问

在这里插入图片描述

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

2.基类和派生类对象赋值转换

  1. 派生类对象可以赋值给基类的对象 / 基类的指针 / 基类的引用。这里有个形象的说法叫切片或者切割。寓意把派生类中父类那部分切来赋值过去
class Person
{
protected:string _name; // 姓名string _sex;// 性别int _age; // 年龄
};
class Student : public Person
{
public:int _No; // 学号
};
Student sobj;
// 1.子类对象可以赋值给父类对象/指针/引用
Person pobj = sobj;
Person* pp = &sobj;
Person& rp = sobj;

每一个子类对象都是一个特殊的父类对象
在这里插入图片描述

当派生类对象被赋值给基类对象时会发生。在切片过程中,派生类对象的部分(通常是额外添加的成员变量和方法)会被忽略,只有基类中定义的部分会被复制到基类对象中。因此,派生类特有的成员变量和方法不会出现在基类对象中,就像它们被“切掉”了一样

在代码中:

class Student : public Person
{
public:int _No; // 学号
};
void Test()
{Student sobj;// 1.子类对象可以赋值给父类对象/指针/引用Person pobj = sobj;  // 切片发生在这里Person* pp = &sobj;  // 没有切片,因为 pp 指向的是一个 Student 对象Person& rp = sobj;   // 没有切片,因为 rp 引用的是一个 Student 对象
}
  • 在行 Person pobj = sobj; 中,由于 pobjPerson 类型的对象,sobj(一个 Student 对象)被赋值给 pobj 时,Student 类特有的 _No 成员被“切掉”,不会体现在 pobj 中。因此,pobj 中无法反映出 sobj 的完整状态和行为。

  • 在行 Person* pp = &sobj; 中,pp 是指向 Person 类型的指针,但它实际上指向了派生类 Student 的对象 sobj,没有发生切片,因为指针指向的是完整的 Student 对象。

  • 在行 Person& rp = sobj; 中,rp 是一个引用 Person 类型,它引用了 sobj,同样没有发生切片,因为引用关联的是 sobj 的完整实体。

实际上,在行 Person& rp = sobj; 中,引用 rp 的确是 Person 类型,但它并不导致对象切片。引用实际上并不拥有它所引用的对象,而只是提供另一个名称来访问现有对象。因此,当我们通过基类引用访问派生类对象时,并没有创建新的对象,也没有丢失派生类的任何部分。

在这行代码中:

Person& rp = sobj;

rp 实际上是对 sobj (它是一个 Student 类型的对象)的另一个访问方式。即使 rp 被声明为 Person 类型的引用,它实际引用的还是 sobj 的完整实体(包含 Person 部分和 Student 特有的部分)。但是,通过 rp 只能直接访问 sobj 中由 Person 定义的成员,Student 特有的成员(如 _No)不可以通过 rp 直接访问,除非进行了适当的强制转换

例子:

Person& rp = sobj;
rp._name = "Name";    // 可以访问,因为_name是Person的成员
// rp._No = 123;      // 错误!无法访问,因为_No是Student特有的成员,即使它实际上存在于sobj中

即使我们通过基类引用或指针操作对象,派生类对象的完整信息(所有成员变量和函数)仍然都在内存中,没有丢失。使用引用和指针时不会发生切片

对象切片的问题仅在派生类对象被赋值给另一个基类类型的对象时才会发生,比如当派生类对象被传值给一个基类对象的函数参数,或者通过赋值构造一个新的基类对象。这时候派生类特有的信息实际上会被切割掉并不会出现在新的基类对象中。在使用引用或指针时,这种情况并不会发生

  1. 基类对象不能赋值给派生类对象
  2. 基类的指针或者引用可以通过强制类型转换赋值给派生类的指针或者引用。但是必须是基类的指针是指向派生类对象时才是安全的。这里基类如果是多态类型,可以使用RTTI(Run-Time Type Information)的dynamic_cast 来进行识别后进行安全转换

3.继承中的作用域

  1. 在继承体系中基类和派生类都有独立的作用域
  2. 子类和父类中有同名成员,子类成员将屏蔽父类对同名成员的直接访问,这种情况叫隐藏,也叫重定义。(在子类成员函数中,可以使用 基类::基类成员 显示访问)
class Person
{
protected:string _name = "a"; // 姓名int _num = 111; // 身份证号
};
class Student : public Person
{
public:void Print(){cout << " 姓名:" << _name << endl;cout << " 身份证号:" << Person::_num << endl;cout << " 学号:" << _num << endl;}
protected:int _num = 999; // 学号
};
void Test()
{Student s1;s1.Print();
};

这段代码展示了成员隐藏,以及如何在派生类中访问基类的被隐藏成员的概念。

  • Student 类中,成员函数 Print 试图访问名称为 _num 的成员变量。由于派生类中存在同名成员,派生类的 _num 会隐藏基类的同名成员。

  • 如果在派生类中尝试访问一个被隐藏的基类成员,需要显式地使用类名限定符来指定基类的成员。在 Print 方法中使用 Person::_num 来访问基类 Person 中的 _num 成员。

输出结果将是:

姓名: a
身份证号: 111
学号: 999
  1. 需要注意的是如果是成员函数的隐藏,只需要函数名相同就构成隐藏
class A
{
public:void fun(){cout << "func()" << endl;}
};
class B : public A
{
public:void fun(int i){fun();cout << "func(int i)->" << i << endl;}
};

B中的fun和A中的fun 不是构成重载,因为不是在同一作用域
B中的fun和A中的fun 构成隐藏,成员函数满足函数名相同就构成隐藏

class B : public A
{
public:void fun(int i)  // 接受一个整型参数{fun();  // 编译器将会提示错误:找不到不带参数的 "fun" 函数。cout << "func(int i)->" << i << endl;}
};

在这个代码中,试图调用基类 Afun 函数。然而,由于派生类 B 提供了一个参数不同的版本 fun(int),所以基类 A 中的 fun 函数在派生类 B 的作用域中被隐藏了。C++ 规则规定,如果派生类提供了和基类同名的函数,基类中同名的函数在派生类的作用域就不再可见了

因此,在 B 类的成员函数 fun(int) 中,调用 fun() 试图无参数调用被隐藏的同名函数会无法编译,因为编译器认为我们试图调用 fun(int) 这个版本,但没有提供参数,导致参数不匹配

修复

为了调用基类 Afun 函数,我们必须显式地使用作用域解析运算符 :: 来指明我们想要调用的函数属于基类作用域:

class B : public A
{
public:void fun(int i){A::fun();  // 正确:调用基类 `A` 中的 `fun`cout << "func(int i)->" << i << endl;}
};

这样,当我们在类 Bfun(int i) 函数中调用 A::fun() 时,它将成功地调用基类 A 无参数的 fun 函数,然后输出整型参数 i 的值。

如果你希望在派生类中保留对基类中同名函数的访问能力(不希望隐藏),可以使用 using 声明在派生类中导入基类中的函数:

class B : public A
{
public:using A::fun;void fun(int i){fun();  // 正确:由于 "using A::fun;",此处调用的是基类 `A` 中的 `fun`cout << "func(int i)->" << i << endl;}
};

在实际编程中,为了避免混淆,通常不建议在派生类中使用与基类成员同名的变量。

本节内容到此结束!感谢大家阅读!

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

OSPF实验系列---3.综合实验

OSPF的综合实验 实验拓扑及要求如下 实验分析 1.R4为ISP&#xff0c;进行IP配置&#xff0c;区域0为公网区域&#xff0c;配置IP地址 2.做MGRE&#xff0c;R3为中心站点&#xff0c;形成Hub-Spoke 3.子网划分 4.私网互通&#xff0c;NAT转换 5.做特殊区域&#xff0c;修改hel…

【C++】STL简介

&#x1f525;个人主页&#xff1a; Forcible Bug Maker &#x1f525;专栏&#xff1a; C 目录 前言什么是STL&#xff1f;STL的历史STL的版本STL六大组件STL的优缺点STL的优点&#xff1a;STL的缺点&#xff1a; 如何学习STL结语 前言 本篇博客主要内容&#xff1a;STL简介。…

01.本地工作目录、暂存区、本地仓库三者的工作关系

1.持续集成 1.持续集成CI 让产品可以快速迭代&#xff0c;同时还能保持高质量。 简化工作 2.持续交付 交付 3.持续部署 部署 4.持续集成实现的思路 gitjenkins 5.版本控制系统 1.版本控制系统概述2.Git基本概述3.Git基本命令 2.本地工作目录、暂存区、本地仓库三者的工作关系…

AD如何从外部导入外框或修改外框大小

一、从外部导入外框 1、从cad中导出dxf文件&#xff0c;从AD中导入导出的文件 2、可参考如下参数设置 3、导入确认后&#xff0c;选择外边框线&#xff08;选择一条边的线然后按Tab键可快速选择&#xff09; 4、到设计-板子形状中选择“按照选择对象定义” 5、板子外形已经出来…

数字电路-5路呼叫显示电路和8路抢答器电路

本内容涉及两个电路&#xff0c;分别为5路呼叫显示电路和8路抢答器电路&#xff0c;包含Multisim仿真原文件&#xff0c;为掌握FPGA做个铺垫。紫色文字是超链接&#xff0c;点击自动跳转至相关博文。持续更新&#xff0c;原创不易&#xff01; 目录&#xff1a; 一、5路呼叫显…

前端工程化05-初始前端工程化Node基本介绍安装配置基础知识

6、初始前端工程化 6.1、工程化概述 虽然前几篇我的目录标题写的前端工程化&#xff0c;但是那些东西并不属于前端工程化的内容&#xff0c;更倾向于是js、jq当中的东西&#xff0c;下面我们将接触真正的前端工程化。 前端工程化开发其实现在是离不开一个东西的&#xff0c;…

观察者模式实战:解密最热门的设计模式之一

文章目录 前言一、什么是观察者模式二、Java实现观察者模式2.1 观察者接口2.2 具体观察者2.3 基础发布者2.4 具体发布者2.5 消息发送 三、Spring实现观察者模式3.1 定义事件类3.2 具体观察者3.3 具体发布者3.4 消息发送 总结 前言 随着系统的复杂度变高&#xff0c;我们就会采…

文件与IO基础常识知识

在这里&#xff0c;只介绍理论知识&#xff0c;不介绍代码。 目录 1.IO 1.1.字面概念 1.2.输入输出模型 2.文件 2.1.文件目录 2.2.文件路径 2.3.文件分类 1.IO 为了我们接下来学习的文件IO&#xff0c;所以我们先来认识什么是IO。 1.1.字面概念 &#xff08;1&#x…

本地基于知识库的大模型的使用教程

本地基于知识库的大模型的使用教程 启动 双击 大模型启动.bat文件&#xff0c;内容如下&#xff1a; cmd /k "cd /d G:\Anaconda3\Scripts && activate.bat && cd /d D:\docdb_llm && conda activate python3.11 && python startup.py…

MFC 列表控件删除实例(源码下载)

1、本程序基于前期我的博客文章《MFC下拉菜单打钩图标存取实例&#xff08;源码下载) 》 2、程序功能选中列表控件某一项&#xff0c;删除按钮由禁止变为可用&#xff0c;点击删除按钮&#xff0c;选中的项将删除。 3、首先在主界面添加一个删除参数按钮。 4、在myDlg.cpp 文件…

STM32的TIM输入捕获和PWMI详解

系列文章目录 STM32单片机系列专栏 C语言术语和结构总结专栏 文章目录 1. IC输入捕获 2. 频率测量 3. 主模式、从模式、触发源选择 4. 输入捕获基本结构 5. PWMI模式 6. 代码示例 6.1 PWM.c 6.2 PWM.h 6.3 IC.c 6.4 IC.h 6.5 完整工程文件 输出比较可以看下面这篇…

python报错SyntaxError

如果报这个错&#xff0c; 在你的相应的demo.py文件首行输入下面的&#xff0c;可以多试一下&#xff0c;之后就好了。 这个解决方法也是参考其他大佬的做法&#xff0c;不知道为什么python中#是注释&#xff0c;这个也会起作用。 然后就神奇的发现问题解决了。发现下面的代码…

window系统安装MySQL

MySQL的安装和配置 根据不同的系统平台&#xff0c;MySQL由不同安装方式和安装包。 官方下载对应的安装包 官网&#xff1a;www.mysql.com 下载地址&#xff1a;MySQL :: Download MySQL Community Server (Archived Versions) window系统 一、安装包&#xff08;Windows…

进位计数制

目录 前言 最古老的计数方法 十进制数 推广:r 进制计数法 任意进制->十进制 二进制<->八进制,十六进制 十进制->任意进制 真值和机器数 总结: 前言 本篇文章我们正式进入第二章:数据的表示和运算,通过第一章的学习我们知道了现代计算机的结构 那数据如何…

华为OD机试 - 会议室占用时间段(Java 2024 C卷 100分)

华为OD机试 2024C卷题库疯狂收录中&#xff0c;刷题点这里 专栏导读 本专栏收录于《华为OD机试&#xff08;JAVA&#xff09;真题&#xff08;A卷B卷C卷&#xff09;》。 刷的越多&#xff0c;抽中的概率越大&#xff0c;每一题都有详细的答题思路、详细的代码注释、样例测试…

✔ ★Java项目——设计一个消息队列(五)【虚拟主机设计】

虚拟主机设计 创建 VirtualHost实现构造⽅法和 getter创建交换机删除交换机创建队列删除队列创建绑定删除绑定发布消息 ★路由规则1) 实现 route ⽅法2) 实现 checkRoutingKeyValid3) 实现 checkBindingKeyValid4) 实现 routeTopic5) 匹配规则测试⽤例6) 测试 Router 订阅消息1…

动态规划-子序列问题1

文章目录 1. 最长递增子序列&#xff08;300&#xff09;2. 摆动序列&#xff08;376&#xff09;3. 最长递增子序列的个数&#xff08;673&#xff09;4. 最长数对链&#xff08;646&#xff09; 1. 最长递增子序列&#xff08;300&#xff09; 题目描述&#xff1a; 状态表…

38.基础乐理-其余调号说明

目前只写了自然大调&#xff0c;还有其它的调式没有写&#xff0c;大调中还有 和声大调 与 旋律大调&#xff0c;除了大调&#xff0c;还有小调式、五声调式、中古调式等还有很多很多&#xff0c;这些东西是需要对于调号、拍号&#xff0c;对于五线谱、对于音程和弦都有一定程度…

OS考研chapter3内存管理

目录 一、基础知识点补充 1.内存、内存地址概念与联系 2.按byte编址 vs 按字编码 二、进程运行的基本原理 1.指令的工作原理 2.逻辑地址 vs 物理地址 3.从写程序到程序运行 &#xff08;1&#xff09;编辑源代码 &#xff08;2&#xff09;编译 &#xff08;3&#xf…

JZ69跳台阶

&#x1f600;前言 青蛙跳台阶是一个经典的问题&#xff0c;它描述了一只青蛙每次可以跳上1级台阶或者2级台阶&#xff0c;问跳上一个n级的台阶有多少种跳法。这个问题看似简单&#xff0c;实则蕴含了一定的数学思维和递推关系。在本文中&#xff0c;我们将通过分析问题的特性和…