【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.mfbz.cn/a/590437.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

webpack基础---常用loader

webpack 命令式和配置文件 html-webpack-plugin 配置项&#xff1a;{ templete: filename: inject: } 清除上次打包的文件&#xff0c;output: { clear: true } mode选项&#xff1a; none development prodution souce-map&#xff1a;可以精准定位代码行数 { devt…

使用node调用chrome(基于selenium-webdriver包)

下载测试版chrome和chromedriver https://googlechromelabs.github.io/chrome-for-testing/ 把chromedriver复制到chrome的文件里 设置环境变量 编写代码 const { Builder, Browser, By, Key, until } require(selenium-webdriver) const puppeteer require(puppeteer)//查…

Flask模版详解

Flask模版详解 概述Jinja2模板引擎渲染模版的步骤变量控制结构自定义错误页面链接静态文件 概述 模板是一个包含响应文本的文件&#xff0c;其中包含用占位变量表示的动态部分&#xff0c;其具体值只在请求的上下文中才能知道。使用真实值替换变量&#xff0c;再返回最终得到的…

空闲缓冲区(empty) 和 非空缓冲区(full) 的的概念和区别【操作系统 生产者——消费者进程】

首先&#xff0c;我们得有个环境——通常是个缓冲池&#xff0c;这个池子里可以塞很多缓冲区&#xff0c;它们是用来存放数据的。生产者就是那个不停造东西的家伙&#xff0c;而消费者则是等着用这些东西的人。 1. 空闲缓冲区&#xff08;empty&#xff09;&#xff1a; 这玩意…

C语言:文件操作(上)

片头 嗨&#xff01;小伙伴们&#xff0c;今天我们来学习新的知识----文件操作&#xff0c;准备好了吗&#xff1f;我要开始咯! 目录 1. 为什么使用文件&#xff1f; 2. 什么是文件&#xff1f; 3. 二进制文件和文本文件&#xff1f; 4. 文件的打开和关闭 5. 文件顺序读写…

硬盘选购指南

转载请注明出处&#xff01; author karrysmile date 2024年5月3日19:10:52 结论 先给用途分类和价格表 前置知识 没有不好的品牌&#xff0c;只有不好的系列。不用认准哪个品牌就不好&#xff0c;认准口碑好&#xff0c;稳定性好的系列买。&#xff08;杂牌别买&#xff0…

系统架构设计师错题集

在实时操作系统中&#xff0c;两个任务并发执行&#xff0c;一个任务要等待另一个任务发来消息&#xff0c;或建立某个条件后再向前执行&#xff0c;这种制约性合作关系被称为任务的&#xff08;9&#xff09;。 (9)A.同步 B.互斥 C.调度 D.执行 【答案】A 【解析】本题考查…

2024年北京高校后勤餐饮博览会|北京餐饮展览会

高联采高校后勤餐饮博览会 暨第25届北京高校后勤餐饮联合招标采购大会 同期举办&#xff1a;中国北京餐饮供应链博览会 主 题&#xff1a; 因为FOOD校园GOOD / 同创高校大舞台共享精彩高联采 时 间&#xff1a;2024年9月21日-22日 地 点&#xff1a;中国国际展览中心&…

基于深度学习的3D目标检测与跟踪

目标检测和跟踪对于自动驾驶来说是至关重要和基础的任务&#xff0c;旨在从场景中识别和定位出那些预定义类别的对象。在所有形式的自动驾驶数据中&#xff0c;3D点云学习引起了越来越多的关注。目前&#xff0c;有许多用于3D目标检测的深度学习方法。然而&#xff0c;鉴于点云…

Java——数组

一&#xff1a;数组 &#xff08;1&#xff09;数组的定义&#xff08;声明&#xff09;&#xff1a; 数据类型 [ ] 数组名 int [ ] a (比较规范) ; int [ ] a ; int a [ ] &#xff08;我个人常用&#xff09;; &#xff08;2&#xff…

(4)传输层

1.TCP/UDP区别 2.TCP流量控制P60 3.TCP拥塞控制P61 实际曲线尽量接近理想曲线 4.TCP超时重传时间的选择P62 5.TCP可靠传输的实现P63 6.TCP连接管理 建立 释放 7.TCP报文段的首部格式P66

LeetCode题练习与总结:柱状图中最大的矩形--84

一、题目描述 给定 n 个非负整数&#xff0c;用来表示柱状图中各个柱子的高度。每个柱子彼此相邻&#xff0c;且宽度为 1 。 求在该柱状图中&#xff0c;能够勾勒出来的矩形的最大面积。 示例 1: 输入&#xff1a;heights [2,1,5,6,2,3] 输出&#xff1a;10 解释&#xff1a…

【分布式系统】FLP、CAP、BASE、ACID理论简介

分布式系统一致性模型 在说FLP&#xff0c;CAP&#xff0c;BASE&#xff0c;ACID理论前&#xff0c;必须先说说分布式系统的一致性模型&#xff0c;它是其他理论的基础知识。 依次介绍几个相关的概念&#xff1a; 分布式系统是由多个不同的服务节点组成&#xff0c;节点与节…

Java将文件目录转成树结构

在实际开发中经常会遇到返回树形结构的场景&#xff0c;特别是在处理文件系统或者是文件管理系统中。下面就介绍一下怎么将文件路径转成需要的树形结构。 在Java中&#xff0c;将List<String>转换成树状结构&#xff0c;需要定义一个树节点类&#xff08;TreeNode&#…

【linux】初步认识文件系统

初步认识文件系统 前置知识的简单了解简单回顾C语言的文件操作stdin&stdout&stderr 系统文件IOopen函数的返回值文件描述符fd打开文件背后的操作文件描述符的分配规则 前置知识的简单了解 文件包括了文件内容和文件属性两个部分(文件内容顾名思义就是文件里面的数据等…

VBA 读取sheet页中的指定区域数据,生成CSV文件

⏹待生成数据的sheet页 ⏹VBA代码 CreateObject("ADODB.Stream")&#xff1a;Microsoft ActiveX Data Objects (ADO) 库中的一个对象&#xff0c;用来处理文件的读写操作。Application.PathSeparator&#xff1a;系统默认的分隔符。Const startRowNum 4&#xff1a…

OpenCV(四)—— 车牌号识别

本节是车牌识别的最后一部分 —— 车牌字符识别&#xff0c;从一个完整的车牌图片到识别出车牌上的字符大致需要如下几步&#xff1a; 预处理&#xff1a;将车牌图片灰度化、二值化&#xff0c;并去除识别时的干扰因素&#xff0c;比如车牌铆钉字符分割&#xff1a;将整个车牌…

for循环赋值

在for循环内将i赋值给j的问题 for(int i0,ji1;i<5;i){//此时j只会等于1cout<<"i-"<<i<<" j-"<<j<<endl; }如图&#xff1a; 将j放入循环体后没问题 for(int i0;i<5;i){int j i1; cout<<"i-"<<…

关于一个error C2664错误代码的解析

具体错误信息如下所示&#xff1a; error C2664: “osgEarth::UID osgEarth::Util::ShaderFactory::addPreProcessorCallback(osg::Referenced *,std::function<void (std::string &,osg::Referenced *)>)”: 无法将参数 2 从“osgEarth::Util::PbrLightEffect::att…

40 生产者消费者模型

生产者消费者模型 概念 为何要使用生产者消费者模型&#xff0c;这个是用过一个容器解决生产者和消费的强耦合问题。生产者和消费者之间不需要通讯&#xff0c;通过阻塞队列通讯&#xff0c;所以生产者生产完数据之后不用等待消费者处理&#xff0c;直接扔给阻塞队列&#xf…