【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)
因为 s3
是 Student
类型的对象。因此,会先输出 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;
}
以上就是继承相关的知识的整理了,欢迎在评论区留言,觉得这篇博客对你有帮助的,可以点赞收藏关注支持一波~😉
相关文章:

【C++】继承相关(基类与派生类的继承关系以及细节整理)
目录 00.引言 01.继承的定义 02.基类和派生类对象 03.继承中的作用域 04.派生类的默认成员函数 05.友元、静态成员 00.引言 继承是面向对象编程中的一个重要概念,它的作用是创建一个新的类,该类可以从一个已存在的类(父类/基类&#x…...
【Web后端】监听器Listener
1、简介 用来监听Servlet组件对象状态发生变化的组件可以监听的源包括:ServetRequest、HttpSession、ServletContext当监听到事件源状态发生变化时,会有对应的响应行为 2、使用方法 在web.xml文件中配置 <listener> <listener-class>com.coder.util.…...

C/C++ 初级球球大作战练手
效果演示: https://live.csdn.net/v/385490 游戏初始化 #include <stdbool.h> #include<stdio.h> #include<stdlib.h> #include<time.h> #include<graphics.h> #include <algorithm> #include<math.h> #include<mmsy…...
ES6之字符串的扩展
字符串的扩展 关键的扩展点及其示例: Unicode 表示与处理 JavaScript 共有6种方法可以表示一个字符。codePointAtpos:String.fromCodePoint…codePoints: **字符串的遍历 for … of **字符串方法的增强 includessearchString[, position]&…...
微信小程序开发,构建NPM报错解决
报错信息如下: message: NPM packages not found. Please confirm npm packages which need to build are belong to miniprogramRoot directory. Or you may edit project.config.jsons packNpmManually and packNpmRelationList appid: wx7144a5d5411…...

【异常】SpringBoot整合RabbitMQ-发送消息报错
错误信息 reply-code406, reply-textPRECONDITION_FAILED - inequivalent arg ‘x-message-ttl’ for queue ‘hello-queue’ in vhost ‘/lq’: received none but current is the value ‘10000’ of type ‘signedint’, class-id50, method-id10 错误原因 hello-queue这…...

通过钉钉卡片进行工单审批
我们通常通过钉钉机器人来发送通知,提醒审批人名下有待办工单需要处理。这种通知方式仅能提醒审批人到ITSM中处理,审批人需要打开电脑登陆平台处理,我们就考虑是否能有一种方式能够满足移动端审批? 这里我们可以使用ITSM的移动端版…...
C语言中的控制语句:深入解析与案例实践2
五、使用goto实现无条件跳转 虽然goto语句在现代编程实践中往往被避免使用,因为它可能导致代码难以理解和维护,但在某些特定情况下,它仍然可以作为一种解决方案。goto语句允许程序无条件地跳转到代码中指定的标签位置。 案例代码:…...

【ARM Cortex-M 系列 2.1 -- Cortex-M7 Debug system registers】
请阅读【嵌入式开发学习必备专栏】 文章目录 Debug system registers中断控制状态寄存器(ICSR)Debug Halting Control and Status Register, DHCSR Debug 寄存器DCRSR与DCRDRCPU 寄存器读操作CPU 寄存器写操作CPU 寄存器选择CPU 寄存器读写示例 调试故障…...

深入解析C#中的async和await关键字
文章目录 一、异步编程的基本概念及其在C#中的实现二、async关键字的定义及其用法三、await关键字的定义及其用法示例代码:使用async和await编写一个简单的异步程序 四、async和await的优点注意事项 五、C#下async和await中常见问题汇总1. 异步方法中的await调用2. …...

【VUE.js】前端框架——未完成
基于脚手架创建前端工程 环境 当安装node.js时,它本身就携带有npm命令。(-v 查版本号)安装VUE CLI npm i vue/cli -g(全局) 创建 vue create 【project name】 镜像源解决方案 输入创建命令后,提示检查更…...

开源的文件压缩和解压缩软件7-Zip
7-Zip是一款开源的文件压缩和解压缩软件,它支持多种压缩格式,包括自有的7z格式以及常见的ZIP、RAR、TAR等格式。下面是对7-Zip的详细介绍,包括其功能特点、使用方法、优缺点等方面的内容。官网:7-Zip 1. 功能特点: 多…...
搜维尔科技:Patchwork 3D 新功能:爆炸视图动画
Patchwork 3D新功能:爆炸视图动画 为什么使用 Patchwork 3D? Patchwork 3D 软件有助于将 CAD 数据转换成真实感的 3D 模型以用于工业用途。 Patchwork 3D 可轻松进行实时渲染,有助于缩短设计周期,或者让您获得效果逼真的渲染图&…...
Redis——RedLock、Zookeeper及数据库实现分布式锁
在分布式系统中,实现分布式锁是确保数据一致性和防止并发问题的重要手段。以下是使用Redis的RedLock算法、ZooKeeper以及数据库实现分布式锁的基本概念和步骤: 1. Redis的RedLock算法 Redis的RedLock算法是Redis官方推荐的一种分布式锁实现方式&#x…...

OpenSSL自签证书并基于Express搭建Web服务进行SSL/TLS协议分析
OpenSSL自签证书并基于Express搭建Web服务进行SSL/TLS协议分析 起因 最近在学习安全协议,大多数实验都是基于Windows下IIS,或者Linux下nginx搭建的Web服务,搭建环境和编写配置文件比较麻烦。而且我有多个不同环境的设备,折腾起来…...

记录一次 vue2 前端项目整合过程
整合成功效果图 具体说明: 项目A是现在的vue2前端项目,项目B是一个开源的工作流前端,项目后端代码已经整合了,就不多提了。这里主要记录下前端整合的过程和思路。 1、开源工作流里面的功能,拷贝到自己对应的vue2项目里…...

物联网五层架构分析
物联网五层架构分析 随着科技的迅速发展,物联网(IoT)作为日常生活中不可或缺的一部分,已融入人们的生活和工作中。物联网五层架构,包括感知层、网络层、数据层、应用层和业务层,扮演着关键的角色。 感知层 …...

【Java EE】多线程(三)线程状态
📚博客主页:爱敲代码的小杨. ✨专栏:《Java SE语法》 | 《数据结构与算法》 | 《C生万物》 |《MySQL探索之旅》 |《Web世界探险家》 ❤️感谢大家点赞👍🏻收藏⭐评论✍🏻,您的三连就是我持续更…...
FFmpeg常用API与示例(一)—— 工具库篇(av_log、AVDictionary、avio)
工具层 1.av_log 可以设置日志的级别,这个看看名字就明白了,也不用过多的解释。 AV_LOG_PANICAV_LOG_FATALAV_LOG_ERRORAV_LOG_WARNINGAV_LOG_INFOAV_LOG_VERBOSEAV_LOG_DEBUG void test_log() {/ av_register_all();AVFormatContext *pAVFmtCtx NU…...

日志的基本用法
目标 1. 掌握如何设置日志级别 2. 掌握如何设置日志格式 3. 掌握如何将日志信息输出到文件中 1. logging模块 Python中有一个标准库模块logging可以直接记录日志 1.1 基本用法 import logging logging.debug("这是一条调试信息") logging.info("这是一条…...

遍历 Map 类型集合的方法汇总
1 方法一 先用方法 keySet() 获取集合中的所有键。再通过 gey(key) 方法用对应键获取值 import java.util.HashMap; import java.util.Set;public class Test {public static void main(String[] args) {HashMap hashMap new HashMap();hashMap.put("语文",99);has…...

Swift 协议扩展精进之路:解决 CoreData 托管实体子类的类型不匹配问题(下)
概述 在 Swift 开发语言中,各位秃头小码农们可以充分利用语法本身所带来的便利去劈荆斩棘。我们还可以恣意利用泛型、协议关联类型和协议扩展来进一步简化和优化我们复杂的代码需求。 不过,在涉及到多个子类派生于基类进行多态模拟的场景下,…...
工程地质软件市场:发展现状、趋势与策略建议
一、引言 在工程建设领域,准确把握地质条件是确保项目顺利推进和安全运营的关键。工程地质软件作为处理、分析、模拟和展示工程地质数据的重要工具,正发挥着日益重要的作用。它凭借强大的数据处理能力、三维建模功能、空间分析工具和可视化展示手段&…...

第 86 场周赛:矩阵中的幻方、钥匙和房间、将数组拆分成斐波那契序列、猜猜这个单词
Q1、[中等] 矩阵中的幻方 1、题目描述 3 x 3 的幻方是一个填充有 从 1 到 9 的不同数字的 3 x 3 矩阵,其中每行,每列以及两条对角线上的各数之和都相等。 给定一个由整数组成的row x col 的 grid,其中有多少个 3 3 的 “幻方” 子矩阵&am…...

ArcGIS Pro制作水平横向图例+多级标注
今天介绍下载ArcGIS Pro中如何设置水平横向图例。 之前我们介绍了ArcGIS的横向图例制作:ArcGIS横向、多列图例、顺序重排、符号居中、批量更改图例符号等等(ArcGIS出图图例8大技巧),那这次我们看看ArcGIS Pro如何更加快捷的操作。…...
是否存在路径(FIFOBB算法)
题目描述 一个具有 n 个顶点e条边的无向图,该图顶点的编号依次为0到n-1且不存在顶点与自身相连的边。请使用FIFOBB算法编写程序,确定是否存在从顶点 source到顶点 destination的路径。 输入 第一行两个整数,分别表示n 和 e 的值(1…...

HashMap中的put方法执行流程(流程图)
1 put操作整体流程 HashMap 的 put 操作是其最核心的功能之一。在 JDK 1.8 及以后版本中,其主要逻辑封装在 putVal 这个内部方法中。整个过程大致如下: 初始判断与哈希计算: 首先,putVal 方法会检查当前的 table(也就…...

视频行为标注工具BehaviLabel(源码+使用介绍+Windows.Exe版本)
前言: 最近在做行为检测相关的模型,用的是时空图卷积网络(STGCN),但原有kinetic-400数据集数据质量较低,需要进行细粒度的标注,同时粗略搜了下已有开源工具基本都集中于图像分割这块,…...
Go 语言并发编程基础:无缓冲与有缓冲通道
在上一章节中,我们了解了 Channel 的基本用法。本章将重点分析 Go 中通道的两种类型 —— 无缓冲通道与有缓冲通道,它们在并发编程中各具特点和应用场景。 一、通道的基本分类 类型定义形式特点无缓冲通道make(chan T)发送和接收都必须准备好࿰…...

关于easyexcel动态下拉选问题处理
前些日子突然碰到一个问题,说是客户的导入文件模版想支持部分导入内容的下拉选,于是我就找了easyexcel官网寻找解决方案,并没有找到合适的方案,没办法只能自己动手并分享出来,针对Java生成Excel下拉菜单时因选项过多导…...