【C++】———— 继承

 9efbcbc3d25747719da38c01b3fa9b4f.gif

                                                      作者主页:     作者主页

                                                      本篇博客专栏:C++

                                                      创作时间 :2024年7月5日

9efbcbc3d25747719da38c01b3fa9b4f.gif

一、什么是继承?

继承的概念

定义:

继承机制就是面向对象设计中使代码可以复用的重要手段,它允许在程序员保持原有类特性的基础上进行扩展,增加功能。这样产生的新类,称为派生类(子类),被继承的类称为基类(父类)。

继承的定义:

下面我们来看一下实力更深入的了解一下它吧。

下面是一个Student类继承Person类的具体实例:

#include<iostream>

using namespace std;

class Person
{
public:
	void Print()
	{
		cout << _height << endl;
		cout << _age << endl;
		cout << _name << endl;
	}
protected:
	double _height = 1.85;//身高
	int _age = 20;//年龄
	string _name = "zhangyu";//姓名
};
class Student :public Person
{
private:
	int _stuid = 123456;//学号
	int _grade = 1;//年级
};


int main()
{
	Person p;
	Student s;

	return 0;
}

看一下这张图片,就可以看出来,使用Student定义的s,就继承了Person的成员和函数。

当然不同继承方式的继承效果也就不同:

我们先来说一下关于private,对于基类中的private成员,即使派生来对象中,但是语法上限制派生类对象无论在类里面还是在类外面都无法访问它。看一段代码更好的去理解

class Person
{
//public:
	/*void Print()
	{
		cout << _height << endl;
		cout << _age << endl;
		cout << _name << endl;
	}*/
private:
	double _height = 1.85;//身高
	int _age = 20;//年龄
	string _name = "zhangyu";//姓名
};
class Student :public Person
{
public:
	void Print()
	{
		cout << _height << endl;
		cout << _age << endl;
		cout << _name << endl;
		cout << _stuid << endl;
		cout << _grade << endl;
	}

这里我们可以看到对于被private修饰的对象是无法在子类中访问的

所以这里也是private和protected的区别之一。

所以我们这里可以得到以下几个理论:

  1. 积累private成员在派生类中无论以什么方式继承都是不可见的,这里的不可见是指基类的私有成员还是被派生类对象中,但是语法上限制派生类对象不管在类里面还是在类外面都无法去访问他们。
  2. 基类中的private成员在派生类中是不能被访问,如果基类成员不想再类外直接被访问,但需要在派生类中被访问,就用protected定义。可以看出保护成员限定符可能是由于继承才出现的。
  3. 实际我们对上面的表格总结一下可以发现,基类的私有成员在子类中都是不可访问的,基类的其他成员的访问方式==min(成员在基类中的访问限定符,继承方式)。
  4. 使用private默认的继承方式是private,使用struct的默认继承方式是public,但是我们在实际开发中最好还是写出继承方式。
  5. 在实际应用中一般使用public继承,几乎很少使用private和protected,也不提倡使用他们,因为protected继承下来的成员只能在派生类里面使用,实际开发中可维护性不强。

二、基类与派生类的赋值转换:

我们在前面的学习知道相近类型之间是能够赋值,因为他们之间会发生隐式类型转换。

int a = 10;
char b = a;//隐式类型转换
char& c = a;//报错
const char& c = a;//正确

double& d = a;//报错
const double& d = a;

char& c = a; 和 double& d = a; 这两行代码是非法的。因为引用必须绑定到与其类型完全匹配的对象上,否则就会引起权限的放大,因为产生的临时对象具有常性,常性是一种编程中的约束和特性,所以不能将 int 类型的变量直接绑定到 char 引用或 double 引用。

那么基类与派生类直接是否也遵循这个转换规则呢,接下来让我们以Person类与Student类来验证一下吧。

2.1派生类对象赋值给基类对象

派生类对象是可以赋值给基类对象的,因为派生类对象本就存在基类成员。相反,基类成员就无法赋值给派生类成员,因为有些成员派生类有,而基类没有。

所以就会报错

2.2派生类对象的引用赋值给基类对象

派生类对象的引用赋值能够给基类对象,其中引用不许需要const,证明其赋值之间并没有发生隐式类型转换,产生临时对象。

	Student s;
	Person& rp = s;//ok

2.3派生类对象指针赋值给基类对象

派生类对象的指针能够赋值给基类对象,这种情况与引用十分类似。

	Student s;
	Person* pp = &s;//ok

2.4基类指针赋值给派生类指针

基类指针能够通过强转赋值给派生类指针,但是也可能造成越界访问

	Person p;
	Student *sp = (Student*) & p;//ok

最后总结出基类与派生类的赋值转换遵循以下规则:

  1. 派生类对象可以赋值给基类的对象 ,基类的指针,基类的引用。这里有个形象的说法叫切片或者切割。寓意把派生类中父类那部分切来赋值过去。
  2. 基类对象不能赋值给派生类对象。
  3. 基类的指针或者引用可以通过强制类型转换赋值给派生类的指针或者引用。但是必须是基类的指针是指向派生类对象时才是安全的。这里基类如果是多态类型,可以使用RTTI,dynamic_cast 来进行识别后进行安全转换。

三、继承的作用域

3.1同名变量

看这里,我们基类与派生类都定义了height变量,那么这里输出谁呢?

这里我们可以看到输出的结果是1.9,要是想输出基类中的height如何做呢,我们就要加上域作用限定符,

	void Print()//隐藏
	{
		cout <<Person:: _height << endl;
		cout << _age << endl;
		cout << _name << endl;
	}

这样就输出1.85了

3.2同名函数

class Person
{
public:
	void func()
	{
		cout << "func()" << endl;
	}

protected:
	double _height = 1.85;//身高
	int _age = 20;//年龄
	string _name = "zhangyu";//姓名
};

class Student :public Person
{
public:
	void func(int i)
	{
		Person::func();
		cout << "func(int i)->" << i << endl;
	}



private:
	double _height = 1.90;
	int _stuid = 123456;//学号
	int _grade = 1;//年级
};


int main()
{
	Student s;
	s.func(1);

	return 0;
}

首先第一个问题,两个fun函数之间是函数重载还是隐藏的关系?答案当然是隐藏关系,因为函数重载针对的是同一个作用域的函数,而基类与派生类直接作用域不同。

在隐藏关系中,同名函数默认调用的当前作用域的函数,如果想调用其他作用域的函数,则需要使用域作用限定符。

四、派生类的默认成员函数

我们知道在类中有6个默认成员函数,如果不显示定义,编译会自动生成。那么如果在派生类中,这几个成员函数是如何生成的呢?

4.1:

class Person
{
public:
	Person()
		:_name("xzy")
		, _height(1.85)
		,_age(20)

	{
		cout << "Person()" << endl;
	}

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


protected:
	double _height = 1.85;//身高
	int _age = 20;//年龄
	string _name = "zhangyu";//姓名
};

class Student :public Person
{
public:
	Student()
		:_height(1.80)
		,_grade(2)
	{
		cout << "Student()" << endl;
	}

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



private:
	double _height = 1.90;
	int _stuid = 123456;//学号
	int _grade = 1;//年级
};


int main()
{
	
	Student s;
	

	return 0;
}

显而易见:派生类对象在调用构造函数时会先调用基类的构造函数,再调用派生类的构造函数。调用析构函数时会先调用派生类的析构函数,再调用基类的析构函数。

4.2:派生类的构造函数必须调用基类的构造函数初始化基类的那一部分成员。如果基类没有默认的构造函数,则必须在派生类构造函数的初始化列表阶段显示调用。

class Person
{
public:
	Person(const char* name)//没有默认构造
		: _name(name)
	{
	}
	Person(const Person& p)
		: _name(p._name)
	{
	}
protected:
	string _name; // 姓名
};
class Student : public Person
{
public:
	Student(int num, const char* name)
		:_num(num)
		//,_name(name) error
		, Person(name)//正确初始化
	{
		;
	}
protected:
	int _num; //学号
};

4.3编译器会对派生类与基类的析构函数名进行特殊处理,都会被处理成destrutor(),所以派生类与基类的析构函数构成隐藏关系。

	Person(const char* name)//没有默认构造
		: _name(name)
	{
		cout << "Person()" << endl;
	}
	~Person()//析构
	{
		cout << "~Person()" << endl;
	}
	Student(int num, const char* name)//构造
		:_num(num)
		//,_name(name) error
		, Person(name)//正确初始化
	{
		cout << "Student()" << endl;
	}
	~Student()
	{
		//因为构成覆盖关系,所以指定域作用限定符
		Person::~Person();
		cout << "~Student()" << endl;
	}

但是为什么Person的析构函数会多调用一次呢?因为编译器为了保证基类的析构最后调用,所以在调用派生类析构函数之后会自动调用基类的构造函数。所以为了保证调用的正确顺序,派生类的析构函数我们不需要显示定义。


4.4拷贝构造与赋值重载必须调用基类的拷贝构造与赋值重载完成对基类的初始化。

//拷贝构造
Person(const Person& p)
	: _name(p._name)
{
}
//赋值重载
Person& operator=(const Person& p)
{
	if (this != &p)
		_name = p._name;

	return *this;
}
Student(const Student& s)//拷贝构造
	:_num(s._num)
	, Person(s)//派生类赋值给基类
{
	;
}
//赋值重载
Student& operator = (const Student& s)
{
	if (this != &s)
	{
		//加域作用限定,否则发生死循环
		Person::operator =(s);
		_num = s._num;
	}
	return *this;
}

派生类赋值重载调用基类赋值重载时记得加域作用限定符,否则就会发生死循环。

五、继承中的友元与静态成员

5.1. 继承中的友元

友元关系不能继承,也就是说父类的友元不是子类的友元,不能访问子类私有和保护成员。

class Student;//声明
class Person
{
public:
	friend void Display(const Person& p, const Student& s);
protected:
	string _name; // 姓名
};
class Student : public Person
{

protected:
	int _stuNum; // 学号
};
void Display(const Person& p, const Student& s)
{
	cout << p._name << endl;
	cout << s._stuNum << endl;
}
int main()
{
	Person p;
	Student s;
	Display(p, s);
	return 0;
}

5.2. 继承中的静态成员

基类定义了static静态成员,则整个继承体系里面只有一个这样的成员,静态成员被所有类对象包括起子类和子类的子类共享。无论派生出多少个子类,都只有一个static成员实例 。

我们可以通过下面这段代码验证:

class Person
{
public:
	Person() { ++_count; }
	string _name; // 姓名
	static int _count; // 统计人的个数。
};
int Person::_count = 0;//静态成员初始化
class Student : public Person
{
protected:
	int _stuNum; // 学号
};
class Graduate : public Student
{
protected:
	string _seminarCourse; // 研究科目
};
int main()
{
	Person p;
	Student s;
	Graduate g;
	cout << &(p._name) << endl;
	cout << &(s._name) << endl;
	cout << &(g._name) << endl;
	cout << &(p._count) << endl;
	cout << &(s._count) << endl;
	cout << &(g._count) << endl;
	return 0;
}

从上图我们就可以看出非静态成员在不同基类与派生类中地址不同,这就说明他们在不同类是独立存在的。而非静态成员却恰恰相反,地址相同,证明基类与派生类都是用同一个静态成员。

六、菱形继承和虚拟继承

6.1. 菱形继承

单继承:一个子类只有一个直接父类的继承关系为单继承

多继承:一个子类有两个或以上直接父类。

菱形继承:就是继承关系近似呈一个菱形形状,如下图:

菱形继承会造成两个问题:数据冗余二义性

class Person
{
public:
	string _name; // 姓名
};
class Student : public Person
{
protected:
	int _num; //学号
};
class Teacher : public Person
{
protected:
	int _id; // 职工编号
};
class Assistant : public Student, public Teacher
{
protected:
	string _majorCourse; // 主修课程
};
void Test()
{
	Assistant a;
	// a._name = "peter"; 这样会产生二义性无法明确知道访问的是哪一个类
	// 需要显示指定访问哪个父类的成员可以解决二义性问题,但是数据冗余问题无法解决
	a.Student::_name = "xxx";
	a.Teacher::_name = "yyy";
}

6.2. 虚拟继承

为了解决数据二义性与冗余的问题,C++引入虚拟继承。虚拟继承用法十分简单,直接在继承前加上一个关键字:virtual

class Person
{
public:
	string _name; // 姓名
};
//虚继承
class Student : virtual public Person
{
protected:
	int _num; //学号
};
//虚继承 
class Teacher : virtual public Person
{
protected:
	int _id; // 职工编号
};
class Assistant : public Student, public Teacher
{
protected:
	string _majorCourse; // 主修课程
};
void Test()
{
	Assistant a;
	 a._name = "peter"; 
}

七、继承与组合

8.1. is-a关系

public继承是一种is-a的关系。也就是说每个派生类对象都是一个基类对象。基类的内部细节对子类可见。

class A
{};

class B : public A
{};

继承允许你根据基类的实现来定义派生类的实现。这种通过生成派生类的复用通常被称为**白箱复用(white - box reuse)。**术语“白箱”是相对可视性而言:在继承方式中,基类的内部细节对子类可见 。继承一定程度破坏了基类的封装,基类的改变,对派生类有很大的影响。派生类和基类间的依赖关系很强,耦合度高。

8.2. has-a关系

组合是一种has - a的关系。假设B组合了A,每个B对象中都有一个A对象。

class A
{};

class B
{
	A _aa;
};

对象组合是类继承之外的另一种复用选择。新的更复杂的功能可以通过组装或组合对象来获得。对象组合要求被组合的对象具有良好定义的接口。这种复用风格被称为黑箱复用(black - box reuse),因为对象的内部细节是不可见的。对象只以“黑箱”的形式出现。组合类之间没有很强的依赖关系,耦合度低。

所以一般推荐优先使用对象组合,而不是类继承 

最后:

十分感谢你可以耐着性子把它读完和我可以坚持写到这里,送几句话,对你,也对我:

1.一个冷知识:
屏蔽力是一个人最顶级的能力,任何消耗你的人和事,多看一眼都是你的不对。

2.你不用变得很外向,内向挺好的,但需要你发言的时候,一定要勇敢。
正所谓:君子可内敛不可懦弱,面不公可起而论之。

3.成年人的世界,只筛选,不教育。

4.自律不是6点起床,7点准时学习,而是不管别人怎么说怎么看,你也会坚持去做,绝不打乱自己的节奏,是一种自我的恒心。

5.你开始炫耀自己,往往都是灾难的开始,就像老子在《道德经》里写到:光而不耀,静水流深。

最后如果觉得我写的还不错,请不要忘记点赞✌,收藏✌,加关注✌哦(。・ω・。)

愿我们一起加油,奔向更美好的未来,愿我们从懵懵懂懂的一枚菜鸟逐渐成为大佬。加油,为自己点赞!

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

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

相关文章

[工具教程]-31-解决mac扣盖后电池耗电快(谁在偷偷的用电池)

查看耗电模式 $ pmset -g查看 hibernatemode 这一行&#xff0c;如果 hibernatemode 后面的数字是 0 &#xff0c;那这种休眠模式下&#xff0c;掉电程度就是非常严重&#xff0c;如果 hibernatemode 后面的数字是 3 &#xff08;大部分人的电脑应该是这个休眠模式&#xff09…

LabVIEW中自定义Ring控件的图标

在LabVIEW中&#xff0c;自定义Ring控件的图标可以让用户界面更加直观和友好。以下是如何在LabVIEW中自定义Ring控件的图标的详细步骤&#xff1a; 步骤1&#xff1a;创建或获取图标 首先&#xff0c;你需要创建或获取你想要在Ring控件中使用的图标。你可以使用图像编辑软件&…

springboot+vue+mybatis图书销售管理系统+PPT+论文+讲解+售后

在Internet高速发展的今天&#xff0c;我们生活的各个领域都涉及到计算机的应用&#xff0c;其中包括图书销售管理系统的网络应用&#xff0c;在外国图书销售管理系统已经是很普遍的方式&#xff0c;不过国内的管理网站可能还处于起步阶段。图书销售管理系统具有网上图书信息管…

BaseServlet的封装

创建BaseServlet的必要性 如果不创建BaseServlet&#xff0c;现在我们只要实现一个功能&#xff0c;我们就需要创建一个servlet! 例如:用户模块(登录&#xff0c;注册&#xff0c;退出录&#xff0c;激活&#xff0c;发送邮件等等功能) 也就是说&#xff0c;我们必须要创建一…

ASO优化不仅仅是苹果商店,安卓商店同样不可忽视

大家一谈起ASO优化&#xff0c;不少人反应大多数都是IOS市场的优化&#xff0c;其实安卓也是不可分割的大市场&#xff0c;在国内手机应用市场&#xff0c;安卓的用户质量在稳步提高&#xff0c;因此开发者也越来越重视安卓市场的推广。做好安卓ASO也是非常必要的。 一、安卓市…

音视频质量评判标准

一、实时通信延时指标 通过图中表格可以看到&#xff0c;如果端到端延迟在200ms以内&#xff0c;说明整个通话是优质的&#xff0c;通话效果就像大家在同一个房间里聊天一样&#xff1b;300ms以内&#xff0c;大多数人很满意&#xff0c;400ms以内&#xff0c;有小部分人可以感…

五行、八卦、天干、地支

零基础入门&#xff1a;五行、八卦、天干、地支 给她家推荐一个小程序&#xff0c;感觉挺准的&#xff01;

PTrade常见问题系列5

回测失败&#xff1a;可用资源不足。 回测运行失败&#xff0c;错误码&#xff1a;2 错误信息&#xff1a;可用资源不足&#xff0c;请稍后在创建。 1、之前客户未限制客户容器使用内存和CPU&#xff0c;周末修改配置&#xff0c;限制了内存和CPU&#xff1b; 2、此报错是用户…

51-3 内网信息收集 - 获取RDP密码信息(没有实验成功)

获取常见应用软件凭据 注意: %USERPROFILE% 是环境变量。在使用系统权限时,可以将 %USERPROFILE% 替换为绝对路径,或使用其他用户的令牌进行操作。 获取 RDP 保存的凭据(远程桌面) 为了避免每次连接服务器都进行身份验证,经常使用 RDP 远程桌面连接远程服务器的用户可能…

STM32蓝牙HID实战:打造低功耗、高性能的客制化键盘

一、项目概述 本项目旨在使用STM32单片机打造一款功能强大的蓝牙客制化键盘&#xff0c;它拥有以下特点&#xff1a; 九键布局&#xff0c;小巧便携: 满足日常使用需求&#xff0c;方便携带。全键可编程: 所有按键和旋钮均可通过电脑软件自定义快捷键&#xff0c;实现个性化功…

融资融券两融技巧,开通条件和费用详情4%

融资融券费用 融资融券有两项费用&#xff0c;分别是利息和交易佣金。一般情况下融资年息是5%&#xff0c;融券年息是9.35%&#xff0c;目前有极少数一部分券商能够做到4.5%&#xff0c;融资20万一天只需要25元。 然后咱们再看看交易佣金&#xff0c;目前市面上万分之3到千分…

【基于R语言群体遗传学】-12-超显性与次显性

欢迎先看前面的博客&#xff0c;再继续进行后面的内容&#xff1a; 群体遗传学_tRNA做科研的博客-CSDN博客 当杂合子的适应度超出纯合子的范围时&#xff0c;二倍体能够展现出更多令人着迷的选择实例。这种形式的一种是杂合子优势&#xff0c;或称为“超显性”&#xff0c;其…

(三)前端javascript中的数据结构之链表上

在js中&#xff0c;没有为我们提供原生的数据结构支持的&#xff0c;但是在java中是有提供的。所以需要我们去模拟这种结构实现。 链表中最关键的一个元素&#xff0c;就是头节点&#xff0c;头节点不存储数据&#xff0c;指向第一个节点链表中几乎所有的操作都要从头结点开始。…

数字流的秩

题目链接 数字流的秩 题目描述 注意点 x < 50000 解答思路 可以使用二叉搜索树存储出现的次数以及数字的出现次数&#xff0c;方便后续统计数字x的秩关键在于构建树的过程&#xff0c;如果树中已经有值为x的节点&#xff0c;需要将该节点对应的数字出现次数加1&#xf…

味全财税数字化实践分享,百望云助力实现数电票管理能力升级!

随着金税四期的启动&#xff0c;数电票全面落地应用&#xff0c;这不仅大幅提高了发票处理效率&#xff0c;更重塑了企业的财务管理流程&#xff0c;为企业财务数字化转型指明了新的方向。 杭州味全食品有限公司&#xff08;以下简称“味全”&#xff09;正在推进财税数字化建设…

(2024,稀疏 MoE,大量小专家,参数高效专家检索 PEER,product key 检索)混合百万专家

Mixture of A Million Experts 公和众与号&#xff1a;EDPJ&#xff08;进 Q 交流群&#xff1a;922230617 或加 VX&#xff1a;CV_EDPJ 进 V 交流群&#xff09; 目录 0. 摘要 1. 简介 2. 方法 3. 实验 0. 摘要 标准 Transformer 架构中的前馈&#xff08;feedforward&a…

轻松集成,高效变现:Flat Ads SDK助力开发者轻松跨越广告变现门槛

在当今的移动应用开发领域,广告变现是开发者们普遍关注的重要话题。如何在不影响用户体验的前提下,最大化地实现广告收益,成为了许多开发者面临的挑战。为此,Flat Ads SDK 应运而生,它以“轻松集成,合规守护,高效变现”为核心理念,帮助开发者轻松解决流量变现难题。 一、高效变…

tauri + vue3 如何实现在一个页面上局部加载外部网页?

&#x1f3c6;本文收录于「Bug调优」专栏&#xff0c;主要记录项目实战过程中的Bug之前因后果及提供真实有效的解决方案&#xff0c;希望能够助你一臂之力&#xff0c;帮你早日登顶实现财富自由&#x1f680;&#xff1b;同时&#xff0c;欢迎大家关注&&收藏&&…

惟客数据Q2荣誉成绩单新鲜出炉

作为众多头部企业客户的数字化经营合作伙伴 WakeData惟客数据一直坚持以客户为中心&#xff0c;以数据驱动 致力于让数据智能商业落地更敏捷 凭借值得信赖的客户经营数字化和资源管理数字化实力 惟客数据在2024年第二季度斩获多项荣誉 1、 第一新声《2023年度中国高科技高…

从入门到精通:Shopee,lazada,temu自养号测评成本、步骤、技巧详解

测评对于卖家来说是一种成本低回报快的推广方式&#xff0c;可以减少高额的平台广告费用&#xff0c;因此是一种很好的辅助手段&#xff0c;对商品的曝光、流量、转化和权重等方面起到了很好的辅助作用 建议还是自己精养一批账号&#xff0c;账号在自己手里比较安全可控&#…