【C++】面向对象三大特性之继承

news2024/12/23 18:16:44

【C++】面向对象三大特性之继承

  • 继承的概念
  • 继承基类成员访问方式的变化
  • 子类到父类对象之间赋值兼容转换
  • 继承中的作用域
  • 子类的默认成员函数
  • 继承和友元、静态成员的关系
  • 菱形继承和菱形的虚拟继承
    • 虚拟继承解决二义性和数据冗余

继承的概念

继承:是面向对象程序设计使代码可以复用的最重要的手段,它允许程序员在保持原有类特性的基础上进行扩展,增加功能,这样产生新的类,称派生类 。而以前我们接触的复用都是函数复用,继承是类设计层次的复用

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

在这里插入图片描述
继承后的成员权限:

  1. 基类private成员在派生类中无论以什么方式继承都是不可见的
  2. 基类的其他成员在子类的访问方式 == Min(成员在基类的访问限定符,继承方式),public > protected> private。
class A {
public:
    int a = 10;
protected:
    int b = 20;
private:
    int c = 30;
};

class B :private A {
    void print()
    {
        cout << a << endl;
        cout << b << endl;
        //cout << c << endl;不可访问
    }
};

子类到父类对象之间赋值兼容转换

  • 派生类对象 可以赋值给 基类的对象 / 基类的指针 / 基类的引用。这里有个形象的说法叫切片或者切割
  • 基类对象不能赋值给派生类对象
  • 基类的指针或者引用可以通过强制类型转换赋值给派生类的指针或者引用。但是必须是基类的指针是指向派生类对象时才是安全的
  • 切片仅限公有继承。如:父类为公有,子类保护或私有继承后,成员变为保护和私有,子类再切片给父类,那么被继承的成员权限与父类不一样了,所以切片仅限公有继承
class A
{
public:
	int _a;
};

class B : private A
{
public:
	int _b;
};
class C : public A
{
public:
	int _c;
};


int main()
{
	B b;
	C c;
	A a(b);//出错
	A aa(c);
}

继承中的作用域

  1. 在继承中父类和子类都有自己独立的类域。
  2. 当子类和父类中存在同名成员时,子类将会屏蔽继承于父类的同名成员,这种情况被称为隐藏或重定义。(子类内部优先使用自己类域的同名成员,子类中使用 基类::基类成员 显示访问)
  3. 子类和父类中的同名成员函数并不构成函数重载,因为它们所处于不同的类域,子类会隐藏父类同名函数。
  4. 父类和子类尽量不要使用同名成员。

子类的默认成员函数

父、子类中成员处理方式

  • 子类中有两部分成员,一类是子类原生的成员,另一类是继承于父类的成员。
  • 对于原生成员,按照普通类调用默认成员函数的规则进行处理;对于继承自父类的成员,将会调用父类中的默认成员函数进行处理。(各玩各的)

需要自己写默认成员函数的情况

  • 对于默认构造函数,如果父类没有默认构造函数,就需要我们自己去写,显式调用构造

  • 对于拷贝构造和赋值重载,如果子类存在深拷贝的问题,这个时候就需要我们自己实现拷贝构造和赋值重载来实现深拷贝

  • 对于析构函数,根据具体实际的情况,如果子类有资源需要释放,那我们就需要自己手动实现析构。

以下是显示写默认成员函数的例子
构造:父类中没有默认构造函数,需要在子类中显示调用
拷贝构造函数:必须调用基类的拷贝构造完成基类的拷贝初始化
派生类的赋值重载必须要调用基类的赋值重载完成基类的复制

class Person
{
public:
	Person(const char* name)
		: _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;
		delete[] p;
	}
protected:
	string _name; // 姓名
	int* p = new int[10];
};
 派生类中
 1、构造函数,父类成员初始化必须调用父类的构造函数
class Student : public Person
{
public:
	Student(const char* name, int num)
		:Person(name)//调用父类的构造函数
		,_num(num)
		, _address("南京")
	{}
	Student(const Student& s)
		:Person(s)//发生切片 子类对象赋值给父类对象
		, _num(s._num)
		, _address(s._address)
	{}
	Student& operator=(const Student& s)
	{
		if (this != &s)
		{
			Person::operator=(s);//切片
			_num = s._num;
		}
		return *this;
	}
	//析构函数
	//子析构函数和父析构函数构成隐藏关系(由于多态关系需求,所有析构函数都会处理成destructor函数名)
	//调用析构函数的时候,先析构子类,在析构父类;子类不需要显示调用父类析构,子类析构后会自动调用父类析构
	~Student()
	{
		cout << "~Student()" << endl;
	}

protected:

	int _num; //学号
	string _address;
};

我们并不需要显式调用父类的析构函数,因为出了子类析构函数的作用域,编译器会自动调用父类的析构。手动调用父类析构将会造成重复析构。

继承和友元、静态成员的关系

  1. 友元关系不能被继承
  2. 父类中的静态成员也会被继承,但是整个继承关系中共用这个静态成员

菱形继承和菱形的虚拟继承

  1. 什么是菱形继承?菱形继承的问题是什么?
    菱形继承是多继承的一种特殊情况,父类对象二义性,数据冗余
    在这里插入图片描述
class A
{
public:
	int _a;
};

class B : public A
//class B : virtual public A
{
public:
	int _b;
};
class C : public A
//class C : virtual public A
{
public:
	int _c;
};

class D : public B, public C
{
public:
	int _d;
};
int main()
{
	D d;
	d.B::_a = 1;
	d.C::_a = 2;
	d._b = 3;
	d._c = 4;
	d._d = 5;
	return 0;
}

在这里插入图片描述
从上图可以看出,D中会有两份A成员,调用时存在二义性和数据冗余

虚拟继承解决二义性和数据冗余

未使用虚拟继承
在这里插入图片描述
通过调用内存,会发现对象d中存在两份的_a,存在二义性和数据冗余

在上述代码中,class B : virtual public A;class C : virtual public A
在这里插入图片描述
当使用了虚拟继承,通过调用内存,发现对象d中仅有一份_a,但是继承于B类和C类的_b和_c上方多了一串地址,查找这个地址,发现这个地址之后的位置存放一个数字0x14(十进制20),这个数字就是b中_a距离公共_a的偏移量,如【0x0115FE5C+0x00000014=0x0115FE70】这样就解决了菱形继承成员冗余的问题

这里的A类叫做虚基类,在对象d中,将虚基类的成员放到一个公共的位置,继承的B、C类需要找到A的成员,通过虚基表【存储偏移量的表】中的偏移量进行计算。可以看到虚基表中第一行还空置了4个字节,这块空间存放的也是一个偏移量,它是指向虚基表的指针或者偏移量表格指针

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

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

相关文章

解析Android VNDK/VSDK Snapshot编译框架

1.背景 背景一&#xff1a; 为解决Android版本碎片化问题&#xff0c;引入Treble架构&#xff0c;它提供了稳定的新SoC供应商接口&#xff0c;引入HAL 接口定义语言&#xff08;HIDL/Stable AIDL&#xff0c;技术栈依然是Binder)&#xff0c;它指定了 vendor HAL 和system fr…

动态规划01背包之416分割等和子集(第10道)

题目&#xff1a; 给你一个 只包含正整数 的 非空 数组 nums 。请你判断是否可以将这个数组分割成两个子集&#xff0c;使得两个子集的元素和相等。 示例&#xff1a; 解法&#xff1a; 先复习一下01背包问题&#xff1a; dp[i][j]的含义&#xff1a;从下标为[0-i]的物品里…

【Spring Boot】Spring Boot的系统配置 — 日志配置

日志配置 日志对于系统监控、故障定位非常重要&#xff0c;比如当生产系统发生问题时&#xff0c;完整清晰的日志记录有助于快速定位问题。接下来介绍Spring Boot对日志的支持。 1.Spring Boot日志简介 Spring Boot自带spring-boot-starter-logging库实现系统日志功能&#…

基于linux下的高并发服务器开发(第一章)- 目录操作函数

09 / 目录操作函数 &#xff08;1&#xff09;int mkdir(const char* pathname,mode_t mode); #include <sys/stat.h> #include <sys/types.h>int mkdir(const char *pathname, mode_t mode); 作用&#xff1a;创建一个目录 参数&#xff1a; pat…

固态硬盘SSD选型测试大纲

一&#xff0c;前言 目前不仅仅是家用电脑系统盘很多都采用了固态硬盘&#xff0c;很多工业产品也选用固态硬盘作为存储介质&#xff0c;这主要得益于固态硬盘相对于机械硬盘的优势。 固态硬盘(Solid State Disk)都是由主控芯片和闪存芯片组成&#xff0c;简单来说就是用固态电…

Python编程从入门到实践_5-10 检查用户名_答案

#《Python编程从入门到实践》&#xff0c;动手试一试&#xff0c;5-10检查用户名&#xff0c;答案。2023-07-15,by qs。 current_users [AaA,bBb,CcC,DdD,EeE] new_users [AAA,bbb,abc,def,hij] for new_user in new_users:current_users_1 []for current_user in current_u…

安达发|汽车零部件行业追溯系统的应用

汽车行业正处于一个蓬勃发展的阶段&#xff0c;随着客户需求的不断变化&#xff0c;生产厂商推出新款商品的速度也越来越快&#xff0c;新项目和变更的不断出现&#xff0c;就可能导致在交付的产品质量方面遇到各种各样的问题。如果这些质量问题得不到及时有效地追溯和控制&…

华为模拟器eNSP过程中所遇问题(40错误)与解决办法

1. 版本 2.打开ensp开启AR2204&#xff0c;报错40 3.弹出文档&#xff0c;挨着试一遍先 安装eNSP的PC上是否存在名为“VirtualBox Host-Only Network”的虚拟网卡 需要启用。虚拟网卡的设置是否符合以下要求&#xff1a;IP地址为192.168.56.1&#xff0c;子网掩码为255.255.2…

Typora设置Gitee图床,自动上传图片

之前写了一篇同类型文章&#xff1a;如何将Typora中图片上传到csdn 实现了Typora本地编辑的内容中的图片&#xff0c;可以直接复制到csdn上进行发布。但是在使用过程中发现sm.ms这个图床站不是很稳定&#xff0c;即使用了翻墙也不稳定。 这篇文章推荐使用Gitee作为图床&#xf…

C++教程(六)——数组

1 数组 1.1 概述 所谓数组&#xff0c;就是一个集合&#xff0c;里面存放了相同类型的数据元素 **特点1&#xff1a;**数组中的每个数据元素都是相同的数据类型 **特点2&#xff1a;**数组是由连续的内存位置组成的 12 一维数组 1.2.1 一维数组定义方式 一维数组定义的三…

FL Studio21编曲软件中文版如何下载更新?

国人习惯称之为水果&#xff0c;也是我个人现在在用的软件。FL Studio是一款比较全面的编曲软件&#xff0c;其通道机架可以使用户添加各种音频采样&#xff0c;快捷编辑节奏型。对于音频的剪辑、拼接、效果处理也非常优秀。非常适合电子音乐编曲以及一些Hiphop。但是其录音、以…

组合数学相关知识总结(目前主要总结了卡特兰数)

全排列 例子&#xff1a; n n n 个数取 m m m 个数有序排放 通项公式&#xff1a; A n m ( P n m ) n ∗ ( n − 1 ) ∗ ( n − 2 ) ∗ ⋅ ⋅ ⋅ ∗ ( n − m 1 ) n ! ( n − m ) ! A_n^m(P_n^m)n*(n-1)*(n-2)**(n-m1) \frac{n!}{(n-m)!} Anm​(Pnm​)n∗(n−1)∗(n−2)∗…

带你【玩转Linux命令】➾ diffstat file 每天2个day05

带你【玩转Linux命令】➾ diffstat & file 每天2个day05 &#x1f53b; 一、文件管理命令1.1 diffstat -根据diff的比较结果&#xff0c;显示统计数字1.2 file-辨识文件类型 &#x1f53b; 总结—温故知新 &#x1f448;【上一篇】 &#x1f496;The Begin&#x1f496; …

几种生成css背景图的方式

几种生成css背景图的方式 使用 CSS 向网页添加背景图案可以为我们的网站增添一些更强烈的个人风格。本文将为介绍几种使用 CSS 生成背景和图案的方式。 Hero Patterns Hero Patterns 是一个免费且易于使用的css背景生成网站&#xff0c;该生成器具有多种来自自然、动物等的图…

蓝牙音视频控制协议(AVCTP)介绍

零.声明 本专栏文章我们会以连载的方式持续更新&#xff0c;本专栏计划更新内容如下&#xff1a; 第一篇:蓝牙综合介绍 &#xff0c;主要介绍蓝牙的一些概念&#xff0c;产生背景&#xff0c;发展轨迹&#xff0c;市面蓝牙介绍&#xff0c;以及蓝牙开发板介绍。 第二篇:Trans…

【手把手】深挖IO(补充篇)

在上一篇【深挖 IO】中已经将各大主流的 IO 网络模型介绍完了&#xff08;还没看过的小伙伴墙裂推荐去瞅一眼 → https://blog.csdn.net/FeenixOne/article/details/129157665 → 不然对这篇的内容可能会有那么一乃乃的影响&#xff09;&#xff0c;那么废话不多说&#xff0c;…

(一)Qt下实现多个海康工业相机内触发采集回调取流显示

系列文章目录 提示&#xff1a;这里是该系列文章的所有文章的目录 第一章&#xff1a;&#xff08;一&#xff09;Qt下实现多个海康工业相机内触发采集回调取流显示 第二章&#xff1a;&#xff08;二&#xff09;Qt下多线程实现多个海康工业相机内触发采集回调取流显示 文章目…

如何设计一个注册中心?(1)概念

1. 为什么需要注册中心 一个集群中有众多服务&#xff0c;每个服务有N个实例&#xff0c;因此需要一个第三方节点来存放每个服务的信息&#xff0c;这样服务需要其它的服务信息&#xff0c;直接到第三方节点获取就行了。这个第三方的节点&#xff0c;就是注册中心。 2. 注册中…

2023.7.14 智慧芽前端面试总结

整体面试体验感蛮不错的&#xff0c;面试官很愿意与我交流&#xff0c;面试的结尾也给出了一定的学习建议。 由于这家公司主要的业务是做搜索引擎方面&#xff0c;估摸着是个自研。面试注重点主要是偏业务处理和针对工具的使用&#xff0c;还有无障碍阅读英文文章&#xff0c;…

C++进阶—哈希/unordered系列关联式容器/底层结构(一篇文章学习哈希)

目录 0. 前言map/set和unordered_map/unordered_set 1. unordered系列关联式容器 1.1 unordered_map 1.1.2 unordered_map的接口说明 1. unordered_map的构造 2. unordered_map的容量 3. unordered_map的迭代器 4. unordered_map的元素访问 5. unordered_map的查询 6…