【C++】继承相关(基类与派生类的继承关系以及细节整理)

目录

00.引言

01.继承的定义

02.基类和派生类对象

03.继承中的作用域

04.派生类的默认成员函数

05.友元、静态成员


00.引言

继承是面向对象编程中的一个重要概念,它的作用是创建一个新的类,该类可以从一个已存在的类(父类/基类)继承属性和方法。

通过下面一个例子可以理解继承的作用:

class teacher
{
private:char _sex;int _age;char _name;
private:int _jobid; // 工号
};class student
{
private:char _sex;int _age;char _name;
private:int _stuid; // 学号
};

在创建学校学校成员信息的时候,需要定义不同的两个类分别用于存放老师和学生的信息种类,但是两者的信息类别是存在重复的(性别、年龄、姓名) ,定义两个类还好,但是如果定义需要更多的类呢,那样就会进行过多的重复工作。此时就可以用继承的方法,将同样的信息类别在父类中定义:

class people
{
public:char _sex;int _age;char _name;
};

使用正确的继承语法,继承父类person的属性:

class teacher : public people
{
public:teacher(): _sex('男') // 在成员初始化列表进行初始化 报错, _age(40), _name('张'){}private:int _jobid; // 工号
};class student : public people
{
public:student(){_sex = '男';_age = 20;_name = '李';}
private:int _stuid; // 学号
};

这里注意:在定义默认构造函数时,不能够通过初始化列表初始化其父类的成员,因为初始化列表只能初始化其自身的成员,所以 teacher() 函数会报错。

01.继承的定义

如图:

我们可以看到,student是子类,也叫派生类,他继承的是父类people,也叫基类,而public是继承方式,继承方式也可以是protected、private,和访问限定符是一样的。

使用不同的继承方式继承的基类成员的访问权限是不一样的,具体关系可以看下面这张表:

可以看出基类的私有成员在派生类中是不可见的,而其他成员的访问方式取决于基类中成员的访问限定符和继承方式。通常情况下,使用 public 继承是最常见的方式,因为它保留了基类的接口,并且派生类可以访问基类的公共和受保护成员。 

注意:

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

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

02.基类和派生类对象

当一个派生类对象被赋值给一个基类对象时,如果使用的是赋值操作符 = 或者拷贝构造函数,那么只会复制派生类对象中基类部分的内容,而派生类特有的成员会被截断

这种行为称为对象切片,因为派生类对象被“切片”成了基类对象

基类的指针或者引用可以通过强制类型转换赋值给派生类的指针或者引用。但是必须是基类的指针是指向派生类对象时才是安全的。

class people
{
public:char _sex;int _age;char _name;
};class student : public people
{
public:student(){_sex = '男';_age = 20;_name = '李';}
public:int _stuid; // 学号
};int main()
{student s1;s1._stuid; // 可以访问people p1 = s1;p1._stuid; // 无法访问return 0;
}

这里将派生类对象s1赋值给基类对象p1后, p1就无法访问s1中的_stuid元素,因为s1是被切片赋值给p1的,如图:

下面的示例用来解释基类的指针赋值:

class Person
{protected :string _name; // 姓名string _sex;  // 性别int _age;    // 年龄
};
class Student : public Person
{
public:int _No; // 学号
};
void Test ()
{Student sobj;// 1.子类对象可以赋值给父类对象/指针/引用Person pobj = sobj;Person* pp = &sobj;Person& rp = sobj;//2.基类对象不能赋值给派生类对象sobj = pobj;//3.基类的指针可以通过强制类型转换赋值给派生类的指针pp = &sobjStudent* ps1 = (Student*)pp; // 这种情况转换时可以的。ps1->_No = 10;pp = &pobj;Student* ps2 = (Student*)pp; // 这种情况转换时虽然可以,但是会存在越界访问的问ps2->_No = 10;
}

03.继承中的作用域

在继承体系中,基类和派生类都有独立的作用域。

访问限定符决定了类的成员在类内部和外部的可见性可访问性。继承方法决定了基类的的成员继承到派生类之后的作用域。

比如用public方法继承基类的成员,那么就在派生类的内部和外部都可以调用基类的成员,但是如果派生类内部也定义了一个和基类名字相同的成员呢?就像这样:

class Person
{
protected:string _name = "小李子"; // 姓名int _num = 111;// 身份证号
};
class Student : public Person
{
public:void Print(){cout << " 姓名:" << _name << endl;cout << " 身份证号:" << Person::_num << endl;cout << " 学号:" << _num << endl;}
protected:int _num = 999; // 学号
};

此时派生类Student和基类Person都定义了成员变量_num,此时在派生类内部调用_num,会默认调用派生类本身的,而基类的_num会被隐藏掉,因为,当子类和父类中有同名成员,子类成员将屏蔽父类对同名成员的直接访问,这种情况叫隐藏,也叫重定义。但是在子类成员函数中,也可以使用 基类::基类成员 显示访问。

同名成员的定义降低了代码的可读性,所以我们在实际继承体系中最好不要定义同名的成员,上述的代码就可以这么修改:

class Person
{
protected:string _name = "小李子"; // 姓名int _IDnum = 111;// 身份证号
};
class Student : public Person
{
public:void Print(){cout << " 姓名:" << _name << endl;cout << " 身份证号:" << _IDnum << endl;cout << " 学号:" << _STnum << endl;}
protected:int _STnum = 999; // 学号
};

04.派生类的默认成员函数

 首先我们要知道默认成员函数是什么:默认成员函数提供默认行为,包括默认构造函数、拷贝构造函数、运算符重载等,这样即使我们没有显式地编写这些函数,类仍然可以正常地进行对象的创建、复制和赋值等操作。那么在派生类中,这几个成员函数是如何生成的呢?

1.默认构造函数

由于派生类继承了基类的成员,在调用派生类的默认构造函数时必须调用基类的默认构造函数初始化基类的那一部分成员:


class Person
{
public:Person(const char* name = "peter"): _name(name){cout << "Person()" << endl;}
protected:string _name; // 姓名
};
class Student : public Person
{
public:Student(const char* name, int num): Person(name), _num(num){cout << "Student()" << endl;}
protected:int _num; //学号
};
int main()
{Student s1("jack", 18);return 0;
}

运行结果:

Person()
Student()

可以看出,编译器先调用基类的默认构造函数后再调用的派生类的默认构造函数。

2.析构函数

派生类的析构函数会在被调用完成后自动调用基类的析构函数清理基类成员。如此一来保证了先清理派生类成员再清理基类成员的顺序

class Person
{
public:Person(const char* name = "peter"): _name(name){cout << "Person()" << endl;}~Person(){cout << "~Person()" << endl;}
protected:string _name; // 姓名
};
class Student : public Person
{
public:Student(const char* name, int num): Person(name), _num(num){cout << "Student()" << endl;}~Student(){cout << "~Student()" << endl;}
protected:int _num; //学号
};
int main()
{Student s1("jack", 18);return 0;
}

运行结果:

Person()
Student()

~Student()
~Person()

因为子类是对基类的继承与延伸,子类受基类的影响,但是基类并不受子类影响,所以在销毁子类对象时,首先释放子类特有的资源,然后再释放基类的资源。

3.拷贝构造函数

派生类的拷贝构造函数需要对两部分进行拷贝构造,一个是派生类自身的成员变量,一个是基类的成员变量,所以派生类的拷贝构造函数必须调用基类的拷贝构造完成基类的拷贝初始化。

class Person
{
public:Person(const Person& p): _name(p._name){cout << "Person(const Person& p)" << endl;}
protected:string _name; // 姓名
};
class Student : public Person
{
public:Student(const Student& s): Person(s), _num(s._num){cout << "Student(const Student& s)" << endl;}
protected:int _num; //学号
};int main()
{Student s1("jack", 18);Student s2(s1);return 0;
}

运行结果:

Person(const Person& p)
Student(const Student& s)

可以看出,也是先调用了基类的拷贝构造函数再调用派生类的。 

4.赋值运算符重载

与拷贝构造类似,派生类的operator=也必须要调用基类的operator=完成基类的复制

class Person
{
public:Person& operator=(const Person& p){cout << "Person operator=(const Person& p)" << endl;if (this != &p)_name = p._name;return *this;}
protected:string _name; // 姓名
};
class Student : public Person
{
public:Student& operator = (const Student& s){cout << "Student& operator= (const Student& s)" << endl;if (this != &s){Person::operator =(s);_num = s._num;}return *this;}
protected:int _num; //学号
};int main()
{Student s1("jack", 18);Student s2(s1);Student s3("rose", 17);s1 = s3;return 0;
}

运行结果:

Student& operator= (const Student& s)
Person operator=(const Person& p)

因为 s3Student 类型的对象。因此,会先输出 Student& operator= (const Student& s)。然后在 Student 类的赋值运算符内部,通过 Person::operator=(s); 调用了基类 Person 的赋值运算符重载函数。

05.友元、静态成员

友元关系不能继承,基类的友元不能访问子类私有和保护成员,但是可以通过将派生类定义成友元的方式,使得派生类也可以访问基类的私有成员

#include <iostream>class Shape {
private:double _width;double _height;public:Shape(double width, double height) : _width(width), _height(height) {}friend class Rectangle; // 将Rectangle类声明为Shape类的友元
};class Rectangle : public Shape {
public:Rectangle(double width, double height) : Shape(width, height) {}void displayArea() {double area = _width * _height; // 可以直接访问基类的私有成员std::cout << "Area of rectangle: " << area << std::endl;}
};int main() {Rectangle rect(5.0, 3.0);rect.displayArea();return 0;
}

在这个例子中,Rectangle 类通过将 Shape 类声明为友元,可以直接访问 Shape 类的私有成员 _width_height

如果基类定义了一个static静态成员,则整个继承体系里面只有一个这样的成员。无论派生出多少类,都只有一个static成员实例

class Person
{
public:Person () {++ _count ;}
protected:string _name ; // 姓名
public:static int _count; // 统计人的个数。
};int Person :: _count = 0;
class Student : public Person
{
protected :int _stuNum ; // 学号
};
class Graduate : public Student
{
protected :string _seminarCourse ; // 研究科目
};
void TestPerson()
{Student s1 ;Student s2 ;Student s3 ;Graduate s4 ;cout <<" 人数 :"<< Person ::_count << endl;Student ::_count = 0;cout <<" 人数 :"<< Person ::_count << endl;
}

以上就是继承相关的知识的整理了,欢迎在评论区留言,觉得这篇博客对你有帮助的,可以点赞收藏关注支持一波~😉

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

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

相关文章

服务攻防——数据库安全

第一步: 端口扫描&#xff1a;nmap 扫不到端口&#xff1a;端口被修改&#xff0c;防护软件&#xff0c;放在内网环境 mysql 内置端口3306 第一种官方漏洞 第一步:先扫描有什么端口开发 用这个错误密码一直访问&#xff0c;最终就进去了 弱口令猜解 不可以直接猜解&#x…

机器人学导论实验1—CoppeliaSim 平台介绍及初步使用BJTU

1. 实验内容分析 对实验内容的理解及关键点&#xff1a; 理解这个实验的关键点在于理解如何使用CoppeliaSim和MATLAB来控制和操作机器人。需要熟悉这两个工具的基本操作&#xff0c;例如如何加载场景、如何修改机器人参数、如何使用MATLAB客户端程序来控制机器人等。此外&#…

Docker 部署 Prometheus 实现一个极简的 QPS 监控

背景 : Prometheus 是近年来最流行的开源监控框架, 其功能强大且易于使用, 拥有各种主流后端语言(Java/Go/Python/Node.js等)与各种场景(如web handler/ k8s/Nginx/MySQL等)的客户端, 并自带图形化显示页面。分享一个快速入门Prometheus 的教程, 实现一个极简的, 后端开发需要特…

Nginx-基础-基础配置-Location

Location 参数匹配模式 参数匹配方式匹配模式说明注意事项精准匹配普通字符串匹配用于标准uri前&#xff0c;要求请求字符串与uri精准匹配&#xff0c;成功则立即处理&#xff0c;nginx停止搜索其他匹配。~正则匹配正则表达式匹配用于正则uri&#xff0c;表示uri包含正则表达…

基于SSM的理发店会员管理系统的设计和实现(有报告)。Javaee项目。ssm项目。

演示视频&#xff1a; 基于SSM的理发店会员管理系统的设计和实现&#xff08;有报告&#xff09;。Javaee项目。ssm项目。 项目介绍&#xff1a; 采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&#xff09;三层体系结构&#xff0…

Docker安装达梦数据库

1.确保已安装Docker 可参考&#xff1a;Linux安装Docker-CSDN博客 2.上传dm镜像并导入安装包 可以从&#xff1a;产品下载 | 达梦数据库下载dm镜像&#xff0c;如下图&#xff1a; docker load -i dm8_20230808.tar 3.导入后查看镜像 docker images 4.启动容器 docker run …

图的概念、性质和存储与简单遍历

前置知识&#xff1a;树的基本概念及性质 为了保证学习效果&#xff0c;请保证已经掌握前置知识之后&#xff0c;再来学习本章节&#xff01;如果在阅读中遇到困难&#xff0c;也可以回到前面章节查阅。 学习目标 掌握图的基本概念掌握图的一些性质 图的概念 基本概念 图 (…

Pytorch如何计算网络参数

方法一. 利用pytorch自身 PyTorch是一个流行的深度学习框架&#xff0c;它允许研究人员和开发者快速构建和训练神经网络。计算一个PyTorch网络的参数量通常涉及两个步骤&#xff1a;确定网络中每个层的参数数量&#xff0c;并将它们加起来得到总数。 以下是在PyTorch中计算网…

如何在 CloudFlare 里屏蔽/拦截某个 IP 或者 IP 地址段

最近除了接的 CloudFlare 代配置订单基本很少折腾自己的 CloudFlare 配置了,今天给大家简单的讲解一下如何在 CloudFlare 里屏蔽/拦截 IP 地址和 IP 地址段,虽然明月一直都很反感针对 IP 的屏蔽拦截,但不得不说有时候还是很有必要的。并且,既然可以拦截屏蔽 IP 自然也可以但…

鸿蒙内核源码分析(VFS篇) | 文件系统和谐共处的基础

基本概念 | 官方定义 VFS&#xff08;Virtual File System&#xff09;是文件系统的虚拟层&#xff0c;它不是一个实际的文件系统&#xff0c;而是一个异构文件系统之上的软件粘合层&#xff0c;为用户提供统一的类Unix文件操作接口。由于不同类型的文件系统接口不统一&#x…

Flink HA模式下JobManager切换时发送告警

资源&版本信息 Flink版本1.14.6 运行平台&#xff1a;K8s HA使用ZK&#xff08;使用K8s的ETC应该是一个道理&#xff09; 详解Flink HA原理 Flink启动时会创建HighAvailabilityServices提供HA和相关基础服务&#xff0c;其中包括leaderRetrievalService和LeaderElecti…

搜索引擎的设计与实现(二)

目录 3 搜索引擎的基本原理 3.1搜索引擎的基本组成及其功能 l.搜索器 (Crawler) 2.索引器(Indexer) 3.检索器(Searcher) 4.用户接口(UserInterface) 3.2搜索引擎的详细工作流程 4 系统分析与设计 4.1系统分析 4.2系统概要设计 4.2系统实现目标 前面内容请移步 搜索引…

宁夏银川市起名专家的老师颜廷利:死神(死亡)并不可怕,可怕的是...

在中国优秀传统文化之中&#xff0c;汉语‘巳’字与‘四’同音&#xff0c;在阿拉伯数字里面&#xff0c;通常用‘4’来表示&#xff1b; 湖南长沙、四川成都、重庆、宁夏银川最靠谱最厉害的起名大师的老师颜廷利教授指出&#xff0c;作为汉语‘九’字&#xff0c;倘若是换一个…

[ACTF新生赛2020]SoulLike

没见过的错误&#xff1a; ida /ctg目录下的hexrays.cfg文件中的MAX_FUNCSIZE64 改为 MAX_FUNCSIZE1024 然后就是一堆数据 反正就是12个字符 from pwn import * flag"actf{" k0 for n in range(12):for i in range(33,127):pprocess("./SoulLike")_flag…

94.二叉树的中序遍历

刷算法题&#xff1a; 第一遍&#xff1a;1.看5分钟&#xff0c;没思路看题解 2.通过题解改进自己的解法&#xff0c;并且要写每行的注释以及自己的思路。 3.思考自己做到了题解的哪一步&#xff0c;下次怎么才能做对(总结方法) 4.整理到自己的自媒体平台。 5.再刷重复的类…

Python爬虫入门:网络世界的宝藏猎人

今天阿佑将带你踏上Python的肩膀&#xff0c;成为一名网络世界的宝藏猎人&#xff01; 文章目录 1. 引言1.1 简述Python在爬虫领域的地位1.2 阐明学习网络基础对爬虫的重要性 2. 背景介绍2.1 Python语言的流行与适用场景2.2 网络通信基础概念及其在数据抓取中的角色 3. Python基…

ssm+vue的公务用车管理智慧云服务监管平台查询统计(有报告)。Javaee项目,ssm vue前后端分离项目

演示视频&#xff1a; ssmvue的公务用车管理智慧云服务监管平台查询统计&#xff08;有报告&#xff09;。Javaee项目&#xff0c;ssm vue前后端分离项目 项目介绍&#xff1a; 采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&…

求阶乘n!末尾0的个数溢出了怎么办

小林最近遇到一个问题&#xff1a;“对于任意给定的一个正整数n&#xff0c;统计其阶乘n&#xff01;的末尾中0的个数”&#xff0c;这个问题究竟该如何解决&#xff1f; 先用n5来解决这个问题。n的阶乘即n!5!5*4*3*2*1120&#xff0c;显然应该为2个数相乘等于10才能得到一个结…

量化交易包含些什么?

我们讲过许多关于量化交易的内容&#xff0c;但是量化交易具体可以做些什么&#xff1f;很多朋友都还不清楚&#xff0c;我们详细来探讨下&#xff01; 第一&#xff1a;什么是量化交易&#xff1f; 量化交易是一种利用先进的数学模型和计算机技术&#xff0c;从大量的历史数…

制造业精益生产KPI和智慧供应链管理方案和实践案例分享

随着工业4.0的推进和国家对制造业高质量发展的重视&#xff0c;工业数据已跃升为生产经营活动中不可或缺的核心要素&#xff0c;同时&#xff0c;工业数据也是形成新质生产力的优质生产要素&#xff0c;助力企业实现高效精益生产。 工业数据在制造业中的作用不可忽视&#xff…