C++之继承(篇一)

news2025/1/11 21:40:50

🍼博客主页:阿博历练记
📖文章专栏:c++
🚍代码仓库:阿博编程日记
🍡欢迎关注:欢迎友友们点赞收藏+关注哦🌹

在这里插入图片描述

文章目录

    • 🥝1.继承的概念及定义
      • 📜1.1继承的概念
      • 🗝️1.2继承图解
      • 🔍1.3代码演示
      • 🔖1.4继承定义
      • (◕‿◕✿)1.5继承方式和访问限定符之间的使用
      • ⭐基类的protected成员和private成员的区别
      • ⭐私有成员和不可见的区别
      • 🔍1.6代码验证
      • ✏️1.7总结
    • 🍆2.基类和派生类对象赋值转换
      • 🔍2.1代码示例
      • ⭐子类赋值给父类对象是隐式类型转换吗?
    • 🍃3.继承中的作用域
      • 🔍代码示例1
      • 🔍代码示例2
    • 🍂4.派生类的默认成员函数
      • 💡4.1派生类怎么显式定义构造函数
      • 💡4.2派生类怎么显式定义拷贝构造函数
      • 💡4.3派生类怎么显式定义赋值重载
      • ⭐4.4派生类怎么显式定义析构函数
      • 🗞️归纳小结

🥝1.继承的概念及定义

📜1.1继承的概念

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

🗝️1.2继承图解

在这里插入图片描述

🔍1.3代码演示

#include <iostream>
using namespace std;
class Person
{
public:
	void Print()
	{
		cout << "name:" << _name << endl;
		cout << "age:" << _age << endl;
	}
protected:
	string _name = "peter"; // 姓名
	int _age = 18;
};
class Student : public Person
{
protected:
	int _stuid; // 学号
};
class Teacher : public Person
{
protected:
	int _jobid; // 工号
};
int main()
{
	Student s;
	Teacher t;
	s.Print();
	t.Print();
	return 0;
}

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

📒友友们,这里我们可以看出s对象和t对象里面都有_name和_age,因为它们两个都继承了Person类,而且它们两个都可以调用print函数,所以这里有个结论友友们要记好哦💰:子类在继承父类的时候不仅继承了成员变量,还有成员函数。

🔖1.4继承定义

在这里插入图片描述

(◕‿◕✿)1.5继承方式和访问限定符之间的使用

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

⭐基类的protected成员和private成员的区别

基类的private成员:无论是哪种继承方式,在子类中或者类外都是不可见的,即不能使用的,我们可以把它抽象的理解为父亲的私房钱🤪,类中类外都不可使用.
基类的protected成员:友友们这里记好,如果此时子类的继承方式是public继承和protected继承,那么继承之后就是子类的protect成员,可以在子类中访问和使用,最重要的是如果当前子类成为父类,那么这个成员就会被新的子类继承下去.如果此时子类的继承是private继承,那么继承之后就会成为子类的private成员,也可以在这个子类中访问和使用,但是如果当前子类成为父类时,这些成员继承下去就无法被新的子类访问和使用了.

⭐私有成员和不可见的区别

私有成员:
①在当前类中可以访问和使用
②在类外不能访问使用
③继承给子类时,无法访问使用
不可见:
①父类的私有成员无论是哪种继承方式子类都不可访问使用
②父类的私有成员继承给子类时,无论是在子类中还是类外都是不可访问使用的

🔍1.6代码验证

在这里插入图片描述

✏️1.7总结

①基类private成员在派生类中无论以什么方式继承都是不可见的。这里的不可见是指基类的私有成员还是被继承到了派生类对象中,但是语法上限制派生类对象不管在类里面还是类外面都不能去访问它
②基类private成员在派生类中是不能被访问,如果基类成员不想在类外直接被访问,但需要在派生类中能访问,就定义为protected。可以看出保护成员限定符是因继承才出现的。
③实际上面的表格我们进行一下总结会发现,基类的私有成员在子类都是不可见。基类的其他成员在子类的访问方式 == Min(成员在基类的访问限定符,继承方式),public > protected > private
④使用关键字class时默认的继承方式是private,使用struct时默认的继承方式是public,不过最好显示的写出继承方式。

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

🔍2.1代码示例

#include <iostream>
using namespace std;
class Person
{
public:
	void Print()
	{
		cout << "name:" << _name << endl;
		cout << "age:" << _age << endl;
	}
	void fun()
	{
		_age++;
	}
protected:
	string _name = "peter"; // 姓名
	int _age = 18;
};
class Student : public Person
{
protected:
	int _stuid; // 学号
};
class Teacher : public Person
{
protected:
	int _jobid; // 工号
};
int main()
{
	Student s;
	Teacher t;
	Person p = s;
	Person& p = s;
	s.Print();
	t.Print();
	return 0;
}

🚦友友们这里一定要注意只有在公有继承的条件下,子类和父类是is-a的关系,子类对象才可以赋值给父类对象,比如杯子是生活用品,这里子类就是杯子,父类就是生活用品。

⭐子类赋值给父类对象是隐式类型转换吗?

这里我们通过代码可以看出来,它们之间没有发生隐式类型转换,因为如果发生转换的话,中间会产生临时变量,而临时变量具有常性,我们在引用的时候需要加const来保证权限平移,而这里我们没有加const,所以不是隐式类型转换。下面就要给友友们引入一个新的概念了——父子类赋值兼容规则📜:在public继承条件下,子类和父类是一个is-a的关系,子类对象赋值给父类对象/父类指针/父类引用时,我们认为是天然的,中间不产生临时对象。这个规则也称之为切割或者切片

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

友友们从这里也可以证明,我们通过让父类对象p去调用fun函数,对_age进行++操作,可以发现s对象内的_age也发生了改变,所以可以证实p就是子类对象中父类那一部分的别名。

🍃3.继承中的作用域

🔍代码示例1

#include <iostream>
using namespace std;
class Person
{
protected:
	string _name = "小李子"; // 姓名
	int _num = 111;  //身份证号
};
class Student : public Person
{
public:
	void Print()
	{

	}
protected:
	int _num = 999; // 学号
};
int  main()
{
	Student s;
	s.Print();
	return 0;
}

很多友友们可能会认为这个代码是错误的,在Student类里面为什么会有两个num变量呢?友友们这里是这样的,这两个num在不同的类域里面,虽然子类继承了父类的num变量,但是父类的num变量是在父类的类域里面定义的,所以这里编译不会出错。

🔍代码示例2

#include <iostream>
using namespace std;
class Person
{
protected:
	string _name = "小李子"; // 姓名
	int _num = 111;  //身份证号
};
class Student : public Person
{
public:
	void Print()
	{
		cout << " 学号:" << _num << endl;
	}
protected:
	int _num = 999; // 学号
};
int  main()
{
	Student s;
	s.Print();
	return 0;
}

这时候友友们可能会好奇如果我们在Student类里面打印_num,那么会是父类的值还是子类的值呢?在这里插入图片描述
很显然是子类中的num变量,友友们这里记住一般编译器采用就近原则,会先在自己的类域里面查找,如果想访问父类中的num变量,我们需要指明类域:Person::_num.在这里插入图片描述
📖总结
💞:父类和子类可以有同名成员,因为它们是独立作用域
💞:默认情况直接访问是子类的,子类同名成员隐藏父类同名成员
💞:继承中,同名的成员函数,函数名相同就构成隐藏,不管参数和返回值,这里注意区分一下函数重载函数重载是在同一作用域,函数名相同,参数不同(参数类型/个数/类型顺序),所以如果我们要通过子类访问父类的同名成员函数时,我们依然需要指明类域:s.Person::fun()

🍂4.派生类的默认成员函数

💡4.1派生类怎么显式定义构造函数

错误案例

#include<vector>
#include<iostream>
using namespace std;
class Person
{
public:
	Person(const char* name = "peter")
		: _name(name)
	{
		cout << "Person()" << endl;
	}

	Person(const Person& p)
		: _name(p._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;
	}

	~Person()
	{
		cout << "~Person()" << endl;
	}
protected:
	string _name; // 姓名
};
class Student : public Person
{
public:
	Student(const char* name, int id)
		: _name(name)
		, _id(id)
	{
		cout << "Student()" << endl;
	}

protected:
	int _id; //学号
};
int main()
{
	Student s;
	return 0;
}

在这里插入图片描述
✔️正确案例

Student(const char* name, int id)
		: Person(name)
		, _id(id)
	{
		cout <<_name <<":"<<_id<< endl;
	}

💡4.2派生类怎么显式定义拷贝构造函数

class Person
{
public:
	Person(const char* name = "peter")
		: _name(name)
	{
		cout << "Person()" << endl;
	}

	Person(const Person& p)
		: _name(p._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;
	}

	~Person()
	{
		cout << "~Person()" << endl;
	}
protected:
	string _name; // 姓名
};
class Student : public Person
{
public:
	Student(const char* name, int id)
		: Person(name)
		, _id(id)
	{
		cout <<_name <<":"<<_id<< endl;
	}
	Student(const Student& s)
		:Person(s)
		,_id(s._id)
	{
		cout << "Student(const Student& s)" << endl;
	}
protected:
	int _id; //学号
};

友友们注意,这里我们在拷贝子类对象中父类那一部分的时候,我们在附用父类的拷贝构造函数的时候很多友友们可能会有疑惑:附用父类的拷贝构造不是应该传父类对象吗,为什么传子类对象s呢?🌠友友们一定不要忘了父子类赋值兼容规则,这里我们虽然传的是子类对象,但其实是经过"切割"把子类中父类的那一部分传了过去🌍。

💡4.3派生类怎么显式定义赋值重载

错误案例

class Person
{
public:
	Person(const char* name = "peter")
		: _name(name)
	{
		cout << "Person()" << endl;
	}

	Person(const Person& p)
		: _name(p._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;
	}

	~Person()
	{
		cout << "~Person()" << endl;
	}
protected:
	string _name; // 姓名
};
class Student : public Person
{
public:
	Student(const char* name, int id)
		: Person(name)
		, _id(id)
	{
		cout <<_name <<":"<<_id<< endl;
	}
	Student(const Student& s)
		:Person(s)
		,_id(s._id)
	{
		cout << "Student(const Student& s)" << endl;
	}
	Student operator=(const Student& s)
	{
		if (&s != this)
		{
			operator=(s);
			_id = s._id;
		}
		return *this;
	}
protected:
	int _id; //学号
};
int main()
{
	Student s("小王",20);
	Student s1(s);
	Student s2("小李",18);
	s2 = s;
	return 0;
}

在这里插入图片描述
✔️正确案例

Student operator=(const Student& s)
	{
		if (&s != this)
		{
			Person::operator=(s);
			_id = s._id;
		}
		return *this;
	}

⭐4.4派生类怎么显式定义析构函数

错误案例1

class Person
{
public:
	Person(const char* name = "peter")
		: _name(name)
	{
		cout << "Person()" << endl;
	}

	Person(const Person& p)
		: _name(p._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;
	}

	~Person()
	{
		cout << "~Person()" << endl;
	}
protected:
	string _name; // 姓名
};
class Student : public Person
{
public:
	Student(const char* name, int id)
		: Person(name)
		, _id(id)
	{
		cout <<_name <<":"<<_id<< endl;
	}
	Student(const Student& s)
		:Person(s)
		,_id(s._id)
	{
		cout << "Student(const Student& s)" << endl;
	}
	Student operator=(const Student& s)
	{
		if (&s != this)
		{
			Person::operator=(s);
			_id = s._id;
		}
		return *this;
	}
	~Student()
	{
		~Person();
		cout << "~Student()" << endl;
	}
protected:
	int _id; //学号
};
int main()
{
	Student s("小王",20);
	Student s1(s);
	Student s2("小李",18);
	s2 = s;
	return 0;
}

在这里插入图片描述
错误案例2

~Student()
	{
		Person::~Person();
		cout << "~Student()" << endl;
	}

在这里插入图片描述

友友们这里我们也可以看出来我们的析构函数每一个对象都多调了一次🙄,为什么会出现这种情况呢?

在这里插入图片描述
✔️正确案例

	~Student()
	{
		cout << "~Student()" << endl;
	}

在这里插入图片描述

🗞️归纳小结

派生类的构造函数必须调用基类的构造函数初始化基类的那一部分成员。如果基类没有默认的构造函数,则必须在派生类构造函数的初始化列表阶段显示调用。
派生类的拷贝构造函数必须调用基类的拷贝构造完成基类的拷贝初始化。
派生类的operator=必须要调用基类的operator=完成基类的复制。
派生类的析构函数会在被调用完成后自动调用基类的析构函数清理基类成员。因为这样才能保证派生类对象先清理派生类成员再清理基类成员的顺序。
派生类对象初始化先调用基类构造再调派生类构造。
派生类对象析构清理先调用派生类析构再调基类的析构。
因为后续一些场景析构函数需要构成重写,重写的条件之一是函数名相同。那么编译器会对析构函数名进行特殊处理,处理成destrutor(),所以父类析构函数不加virtual的情况下,子类析构函数和父类析构函数构成隐藏关系。

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

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

相关文章

基于学校优化算法的无人机航迹规划-附代码

基于学校优化算法的无人机航迹规划 文章目录 基于学校优化算法的无人机航迹规划1.学校优化搜索算法2.无人机飞行环境建模3.无人机航迹规划建模4.实验结果4.1地图创建4.2 航迹规划 5.参考文献6.Matlab代码 摘要&#xff1a;本文主要介绍利用学校优化算法来优化无人机航迹规划。 …

MySQL 分组后统计 TopN 思路优化

一、表信息 表结构如下&#xff1a; CREATE TABLE score (id int(11) NOT NULL AUTO_INCREMENT,name varchar(255) DEFAULT NULL,score int(11) DEFAULT NULL,PRIMARY KEY (id) ) ENGINEInnoDB AUTO_INCREMENT1746687 DEFAULT CHARSETutf8;使用存储过程生成十万条测试数据&am…

Elasticsearch:使用 ES|QL

在我之前的文章 “Elasticsearch&#xff1a;ES|QL 查询语言简介”&#xff0c;我对 ES|QL 做了一个简单的介绍。在今天的文章中&#xff0c;我们来描述如何使用 ES|QL。 REST API 这个用来返回 ES|QL (Elasticsearch qyery language) 的查询结果。它具有如下的格式&#xff1…

MongoDB安全及系例全教程

一、系列文章目录 一、MongoDB安装教程—官方原版 二、MongoDB 使用教程(配置、管理、监控)_linux mongodb 监控 三、MongoDB 基于角色的访问控制 四、MongoDB用户管理 五、MongoDB基础知识详解 六、MongoDB—Indexs 七、MongoDB事务详解 八、MongoDB分片教程 九、Mo…

用扩散AI生成的合成数据的质量评估方法【4个指标】

在线工具推荐&#xff1a; Three.js AI纹理开发包 - YOLO合成数据生成器 - GLTF/GLB在线编辑 - 3D模型格式在线转换 - 3D场景编辑器 为了生成有用的图像数据集&#xff0c;我们使用真实世界的照片数据集作为指南针&#xff0c;探索即时工程的艺术。 我们的稳定扩散&#xff08;…

PPT模板,免费下载!

找免费PPT模板就上这6个网站&#xff0c;各种模板、素材都能找到&#xff0c;质量还很高&#xff0c;赶紧收藏起来&#xff01; 1、菜鸟图库 https://www.sucai999.com/search/ppt/0_0_0_1.html?vNTYwNDUx 菜鸟图库网有非常丰富的免费素材&#xff0c;像设计类、办公类、自媒…

红海云签约中国煤科信息公司,数智引领科技型国企人力资源数字化变革

中煤科工集团信息技术有限公司&#xff08;以下简称“中国煤科信息公司”&#xff09;隶属于中国煤炭科工集团&#xff0c;作为中国煤科核心软件的研发中心、数据技术中心、内部信息化支撑中心&#xff0c;是中国煤科加快智能矿山建设和数字化转型的核心力量。 基于对数字化转…

软件设计模式原则(一)迪米特法则

开一个小专题——详细总结一下软件设计模式原则&#xff0c;这部分在《软计》和《java设计模式》中算是很重要的知识点&#xff0c;值得展开详细讲解一下~首先介绍的是【迪米特法则】 一.定义 迪米特法则又称为最少知识原则&#xff0c;其定义为&#xff1a;一个软件实体应当尽…

vue使用JsBarcode生成条形码

在工作中&#xff0c;有一个需求是接口返回的订单号生成条形码&#xff0c;如图&#xff1a; 1.安装依赖 yarn add jsbarcode2.引入 在script标签中引入 import JsBarcode from jsbarcode 3.使用 this.$refs.a.src的值为条形码的地址。 <template><div><img…

linux下多机器ssh免密码登录配置

20,21,22,23等4台机器配置ssh免密登陆 确认sshd配置 查看/etc/ssh/sshd_config文件&#xff0c;确认如下配置没有被注释掉&#xff1a; AuthorizedKeysFile .ssh/authorized_keys每一台机器修改hosts配置主机名&#xff08;可选&#xff09; 执行ssh命令&#xff0c;如…

积分球测试粉末反射率

积分球测试粉末主要是基于光在积分球内的反射和混合。具体来说&#xff0c;当光线进入积分球时&#xff0c;它将在球的内表面上进行反射。由于积分球的内表面是高反射材料&#xff0c;所以大部分光线将被反射&#xff0c;而不会逃逸出球体。在积分球内&#xff0c;光线经过多次…

基于51单片机的全自动洗衣机系统设计

**单片机设计介绍&#xff0c;基于51单片机的全自动洗衣机系统设计(仿真、程序、论文) 文章目录 一 概要二、功能设计设计思路 三、 软件设计原理图 五、 程序六、 文章目录 一 概要 基于51单片机的全自动洗衣机系统是一种集成控制、传感、显示等功能于一体的智能洗衣机系统&a…

Security ❀ UDP/TCP传输层常见DOS攻击详解

文章目录 1. UDP协议基础2. UDP Flood2.1. 攻击原理2.2. 防护方法 3. TCP三次握手和四次挥手3.1. 三次握手3.2. 四次挥手 4. SYN Flood4.1. 攻击原理4.2. 防护方法 5. SYN-ACK Flood5.1. 攻击原理5.2. 防护方法 6. ACK Flood6.1. 攻击原理6.2. 防护方法 7. FIN/RST Flood7.1. 攻…

来可电子基于UDS的ECU刷写方案

车辆电子控制单元&#xff08;ECU&#xff09;的刷写方式也正在发生重大变化。传统的ECU刷写方法通常使用CAN卡连接电脑进行&#xff0c;现在越来越多的汽车商家和软件开发人员开始采用基于总线UDS来进行ECU刷写。 1、通过我们LKmast上位机软件编写配置刷写步骤 2、导入到我们…

python将图片序列保存成gif

这里用到的模块是imageio。用imageio.mimsave即可将图片序列保存成gif动态图。以下是本人编写的小实验&#xff1a; import cv2 import imageiopaths ["./images/0001.png", "./images/0002.png", "./images/0003.png", ...] frames [] for i…

瑞禧生物分享纳米粉体~二硫化钼粉体 MoS2 纯度:99% 纳米二硫化钼(MoS2)

二硫化钼粉体 名称&#xff1a;二硫化钼粉体 纯度&#xff1a;99% 外观&#xff1a;粉末 纳米二硫化钼(MoS2)粉体硫化钼粉体 二硫化铝化学性质稳定、热稳定性好 、摩擦系数低 、润滑作用优 良且在较为苛 刻的工作环境下能保持 良好的摩擦性能因此二硫化铝被广泛应用于固体润…

一键报警可视对讲管理机10寸触摸屏管理机

一键报警可视对讲管理机10寸触摸屏管理机 一、管理机技术指标&#xff1a; 1、10寸LCD触摸屏&#xff0c;分辨率1024*600&#xff1b; 2、摄像头1200万像素 3、1000M/100M自适应网口&#xff1b; 4、按键设置&#xff1a;报警/呼叫按键&#xff0c;通话/挂机按键&#xff0…

count+group by

一、count()函数 1、count(*) 把所有的行数都查询出来&#xff0c;除非该行中所有的数据为null SELECTCOUNT(*) FROMemployees结果&#xff1a;107 2、count(commission_pct) 把指定列中所有的行数查出来&#xff0c;只要一行为null&#xff0c;那就不计数 SELECTCOUNT(com…

【Qt-22】Qt乱码问题解决

最近在Qt项目中遇到TCP通信接收数据乱码的问题&#xff0c;很是苦恼&#xff0c;经过多次尝试&#xff0c;终于得以解决。 感谢Qt TcpSocket 传递数据乱码显示_qt中socket接受到的客户端数据显示不出来-CSDN博客 彻底解决Qt中文乱码以及汉字编码的问题(UTF-8/GBK)_XX風的博客…