菱形继承的类对父类的初始化、组合、多态、多态的原理等的介绍

news2024/11/17 16:32:54

文章目录

  • 前言
  • 一、菱形继承的类对父类的初始化
  • 二、组合
  • 三、 多态
    • 1. 构成多态
    • 2. 虚函数
    • 3. 虚函数的重写
    • 4. 虚函数重写的两个例外
      • 1. 协变
      • 2. 析构函数的重写
    • 5. C++11 final 和 override
      • 1. final
      • 2. override
    • 6. 设计不想被继承的类
    • 7. 重载、覆盖(重写)、 隐藏(重定义)的对比
  • 四、多态的原理
  • 总结


前言

菱形继承的类对父类的初始化、组合、多态、多态的原理等的介绍


一、菱形继承的类对父类的初始化

#include<iostream>
#include <string>
using namespace std;

class A
{
public:
	A(const char* A)
		:_a(A)
	{
		cout << "class A" << endl;
	}
	string _a;
};

class B : virtual public A
{
public:
	B(const char* A, const  char* B)
		:A(A)
		,_b(B)
	{
		cout << "class B" << endl;
	}
	string _b;
};

class C : virtual public A
{
public:
	C(const char* A, const char* C)
		:A(A)
		,_c(C)
	{
		cout << "class C" << endl;
	}
	string _c;
};

class D : public B, public C
{
public:
	D(const char* A, const char* B, const char* C, const char* D)
		: A(A)
		, B(A, B)
		, C(A, C)
		, _d(D)
	{
		cout << "class D" << endl;
	}
	string _d;
};


int main()
{

	D d("class A", "class B", "class C", "class D");
	
	return 0;
}

结构为:
在这里插入图片描述

  • 因为D类有如上的结构,所以A类会在D类中调用构造函数初始化,并且只调用一次,在D类中初始化B类和C类时,传入的A类不调用A类的构造函数初始化。

在这里插入图片描述

二、组合

  • public继承是一种is-a的关系。也就是说每个派生类对象都是一个基类对象。
  • 组合是一种has-a的关系。假设B组合了A,每个B对象中都有一个A对象。

优先使用对象组合,而不是类继承。

#include <iostream>
using namespace std;

class A
{
protected:
	int _a;
};

class B : public A
{
protected:
	A _bb;
};

int main()
{

	return 0;
}

三、 多态

1. 构成多态

构成多态需要两个条件:

  1. 必须通过基类的指针或者引用调用虚函数
  2. 被调用的函数必须是虚函数,且派生类必须对基类的虚函数进行重写
// 构成多态
#include<iostream>
using namespace std;

class Person
{
public:
	virtual void BuyTicket()
	{
		cout << "购票---全价" << endl;
	}
};

class Student : public Person
{
public:
	virtual void BuyTicket()
	{
		cout << "购票---半价" << endl;
	}
};

//void fun(Person& p)
//{
//	p.BuyTicket();
//}

void fun(Person* p)
{
	p->BuyTicket();
}

int main()
{
	Person p;
	fun(&p);

	Student s;
	fun(&s);

	return 0;
}

在这里插入图片描述

2. 虚函数

虚函数:即被virtual修饰的类成员函数称为虚函数。

class Person
{
public:
	virtual void BuyTicket()
	{
		cout << "购票---全价" << endl;
	}
};

3. 虚函数的重写

虚函数的重写(覆盖):派生类中有一个跟基类完全相同的虚函数(即派生类虚函数与基类虚函数的返回值类型、函数名字、参数列表完全相同),称子类的虚函数重写了基类的虚函数。

在重写基类的虚函数的时候,虽然派生类的虚函数不写virtual也可以构成重写, 但是不建议这样使用

#include<iostream>
using namespace std;

class Person
{
public:
	virtual void BuyTicket()
	{
		cout << "购票---全价" << endl;
	}
};

class Student : public Person
{
public:
	virtual void BuyTicket()
	{
		cout << "购票---半价" << endl;
	}
};

4. 虚函数重写的两个例外

1. 协变

基类与派生类虚函数返回值类型不同。

简单来讲就是 基类与派生类的虚函数的返回值构成父子类指针或引用关系。

#include <iostream>
using namespace std;

class A {};
class B :public A {};

class Person
{
public:
	virtual A* BuyTicket()
	{
		cout << "购票---全价" << endl;
		return new A;
	}
};

class Student : public Person
{
public:
	virtual B* BuyTicket()
	{
		cout << "购票---半价" << endl;
		return new B;
	}
};

void fun(Person& p)
{
	p.BuyTicket();
}


int main()
{
	Person p;
	fun(p);
	Student s;
	fun(s);
	return 0;
}
  • 基类和派生类构成父子类指针的关系

在这里插入图片描述

2. 析构函数的重写

虽然基类和派生类的析构函数的函数名不同,但是编译器会将析构函数的函数名,都处理成destructor,因此可以构成重写。

一般情况下,应该将派生类的析构函数与基类析构函数构成函数重写,使下面的情况delete可以实现多态,保证指向的对象正确调用析构函数。

析构函数可以构成虚函数重写吗, 为什么要构成虚函数重写?

  1. 析构函数加virtual会构成析构函数,因为编译器会将析构函数的名字统一命名为destructor
  2. 构成虚函数重写是因为,我们new一个派生类对象的空间,但是用基类的类型指针接收
  3. 在析构这个对象时,只会进行普通调用,普通调用会按照当前类型, 则只会调用基类的析构函数
  4. 这种情况,我们希望是一个多态调用,按照指向的类型调用析构函数,就需要构成虚函数的重写。
#include <iostream>
using namespace std;


class Person
{
public:
	virtual void BuyTicket()
	{
		cout << "购票---全价" << endl;
	}

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

class Student : public Person
{
public:
	virtual void BuyTicket()
	{
		cout << "购票---半价" << endl;
	}

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

int main()
{
	Person* p1 = new Person;
	Person* p2 = new Student;

	delete p1;
	delete p2; // p2->destuctor() + operator delete(p)

	// 这里我们期望是一个多态调用,而不是普通调用

	p1 = nullptr;
	p2 = nullptr;

	return 0;
}

在这里插入图片描述

5. C++11 final 和 override

1. final

final 修饰的虚函数不能被重写

在这里插入图片描述

final修饰的类,不能被当做基类, 不能被继承

在这里插入图片描述

2. override

检查派生类中的虚函数与基类中虚函数是否构成重写,若不构成重写则报错。

#include <iostream>
using namespace std;


class Person
{
public:
	virtual void BuyTicket()
	{}

};

class Student : public Person
{
public:
	// 检查派生类中的虚函数与基类中的虚函数是否构成重写
	virtual void BuyTicket()override
	{
		cout << "购票---半价" << endl;
	}
};


int main()
{
	Person p;

	return 0;
}

在这里插入图片描述

6. 设计不想被继承的类

将构造函数私有或者将析构函数私有

将构造函数私有

#include <iostream>
using namespace std;

class A
{
public:
	static A CreateObj()
	{
		return A();
	}
private:
	A() {}

};

int main()
{
	A::CreateObj();
	return 0;
}

在这里插入图片描述

7. 重载、覆盖(重写)、 隐藏(重定义)的对比

在这里插入图片描述

四、多态的原理

普通调用在编译时地址就确定了
多态调用在程序运行时,到指向对象的虚函数表中找函数的地址

#include <iostream>
using namespace std;

class Person
{
public:
	virtual void BuyTicket()
	{
		cout << "购票---全价" << endl;
	}

	int _a = 0;

};

class Student: public Person
{
public:
	virtual void BuyTicket()
	{
		cout << "购票---半价" << endl;
	}

	int _b = 1;
};

// 普通调用
//void fun(Person p)
//{
//	p.BuyTicket();
//}

// 多态调用
void fun(Person& p)
{
	p.BuyTicket();
}

int main()
{
	Person p;

	Student s;


	return 0;
}

在这里插入图片描述


总结

菱形继承的类对父类的初始化、组合、多态、多态的原理等的介绍

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

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

相关文章

基于数据挖掘的航空客户满意度分析预测系统

温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长 QQ 名片 :) 1. 项目简介 航空公司致力于提供多样化的服务以满足乘客需求&#xff0c;包括但不限于提供免费无线网络、免费食物饮品、提供网上预约服务、飞机出口位置、座椅舒适度、卫生状况等&#xff0c;并希望以此提升乘…

Linux系统CentOS下挂载磁盘

1. 挂载磁盘步骤总结如下 1. 对磁盘进行分区 2. 对磁盘进行格式化 3. 将磁盘挂载到对应目录 4. 设置开机自动挂载磁盘 2. 对磁盘进行分区 2.1 查看系统设备信息 lsblk指令显示所有块设备信息&#xff1a;显示系统中所有的块设备信息&#xff0c;包括磁盘和分区 lsblk 2…

Mapbox封装图形绘制工具 线,圆,polygon,删除,点 mapbox-gl-draw-circle mapbox-gl-draw

使用插件&#xff0c;安装 npm install mapbox-gl-draw-circle //绘制圆 npm install mapbox/mapbox-gl-draw //绘制点线面删除相关API地址&#xff1a;https://github.com/mohong/mapbox-gl-draw-circle https://github.com/mapbox/mapbox-gl-draw/blob/main/docs/API.md…

Unity 热更新(HybridCLR+Addressable)-设置打包路径和加载路径、打开Hosting服务、打包

四、设置打包和加载路径 五、打开Hosting服务 六、打包 打包完成后路径在Assets同级目录下的ServerData 但是目前没有资源文件对比 修改上面设置后再次打包 里面多了哈希和JSON文件&#xff0c;这俩个就是用于资源对比

dotnet4.0编译问题

因为最近在写cobaltstrike的execute-assembly内存加载的c#项目 用visual studio2022编译&#xff0c;最低net只能用6.0版本的&#xff0c;并且execute-assembly不支持 我想使用4.x版本进行编译&#xff0c;因为visual studio不支持&#xff0c;那么使用命令行进行编译 因为要用…

简单了解Redis(初识阶段)

1.认识Redis 对于Redis有一个很重要的点就是&#xff0c;它存储数据是在内存中存储的。 但是对于单机程序&#xff0c;直接通过变量存储数据的方式是更优的&#xff0c;在分布式系统下 Redis才能发挥威力 因为进程是有隔离性的&#xff0c;Redis可以基于网络&#xff0c;把进…

solidwork怎么隐藏实体再

在实际生产生活中&#xff0c;由于一些零件重叠或覆盖导致我们无法正确装配 我们就需要隐藏实体来看内部结构 假如不需要这个白色的大腿 先双击点他&#xff0c;然后右键 此时即可隐藏 同时右边零件会变白 想重新显示这样操作就可以了

Linux 常用命令(待更新)

1、pwd命令 2、cd命令 3、ls命令 4、locate命令 5、clear命令 6、cat命令 7、head命令 8、tail命令 9、grep命令 10、chmod命令 11、cp命令 12、mv命令 13、mkdir命令 14、rm命令 15、文件压缩和有关归档的命令 16、文件系统的命令 17、与系统管理相关的命令 …

2024 Snap 新款ar眼镜介绍

2024 snap 新款ar眼镜介绍 2024 Snap 新款ar眼镜介绍 助力快速掌握数据集的信息和使用方式。

【中台设计】数字中台,大数据中台解决方案,中台建设指南(资料Word分享)

1. 中台概念 2. 推动企业组织模式演进 3. 建设方法 4 .中台内容 5. 数据安全体系 中台内容围绕数据中台建设评估、整体框架、数据采集&#xff0c;结构化、半结构化、非结构化的数据采集&#xff0c;数据计算能力、存储计算引擎、数据架构、数据挖掘、各种不同数据层建设、模型…

煤矿井下钻场目标检测数据集 5类 voc格式

煤矿井下钻场目标检测数据集 本数据集包含了来自不同钻场和环境背景条件下的70948张图片&#xff0c;涵盖了夹持器、钻机卡盘、煤矿工人、矿井安全帽和钻杆等五类目标&#xff0c;并提供了PASCAL VOC格式的标注文件。 摘要 煤矿井下钻场打钻是解决瓦斯灾害、水害、隐蔽地质灾害…

点云与Open3D

点云数据介绍 点云与三维图像的关系&#xff1a;三维图像是一种特殊的信息表达形式&#xff0c;其特征是表达的空间中三个维度的数据&#xff0c;表现形式包括&#xff1a; 深度图&#xff08;以灰度表达物体与相机的距离&#xff09;&#xff0c;几何模型&#xff08;由CAD软…

Solidity语言:重点学习Solidity编程语言,这是EVM上最常用的智能合约语言。

Solidity是一种面向合约的编程语言&#xff0c;用于在以太坊虚拟机&#xff08;EVM&#xff09;上编写智能合约。它是Solidity开发者在以太坊平台上创建智能合约的主要选择之一。 学习Solidity的重点包括以下几方面&#xff1a; 语法和数据类型&#xff1a;学习Solidity的基本…

工业边缘计算网关和普通网关的区别-天拓四方

随着物联网&#xff08;IoT&#xff09;和工业4.0的快速发展&#xff0c;网关作为连接不同网络和设备的关键设备&#xff0c;其角色和功能日益凸显。在工业环境中&#xff0c;工业边缘计算网关和普通网关虽然都扮演着重要的角色&#xff0c;但它们在功能、应用场景和性能上存在…

算法:69.x的平方根

题目 链接&#xff1a;leetcode链接 思路分析&#xff08;二分算法&#xff09; 当然你可以使用暴力查找&#xff0c;但是二分算法的时间复杂度更好。 我们先用暴力查找找点灵感 x &#xff1a;1 2 3 4 5 6 7 8 x2&#xff1a;1 4 9 16 25 36 49 64 我们的目的是找到一个x…

【Java特性】多态详解——对象类型转换与 instanceof 关键字的运用

多态是指不同类的对象在调用同一个方法时所呈现出的多种不同行为。通常来说&#xff0c;在一个类中定义的属性和方法被其他类继承或重写后&#xff0c;当把子类对象直接赋值给父类引用变量时&#xff0c;相同引用类型的变量调用同一个方法所呈现出的多种不同形态。多态不仅解决…

My_string 运算符重载,My_stack

思维导图 将My_string类中的所有能重载的运算符全部进行重载 、[] 、>、<、、>、<、! 、&#xff08;可以加等一个字符串&#xff0c;也可以加等一个字符&#xff09;、输入输出(<< 、 >>) My_string my_string.h #ifndef MY_STRING_H #define MY_…

【论文】FunAudioLLM:一个旨在增强人类与大型语言模型(LLMs)之间自然语音交互的模型家族

研究背景 1.研究问题&#xff1a;这篇文章要解决的问题是如何增强人类与大型语言模型&#xff08;LLMs&#xff09;之间的自然语音交互。具体来说&#xff0c;研究集中在语音识别、情感识别和音频事件检测&#xff08;多语言&#xff09;以及语音生成&#xff08;多语言、零样…

云栖3天,云原生+ AI 多场联动,新产品、新体验、新探索

云栖3天&#xff0c;云原生 AI 20场主题分享&#xff0c;三展互动&#xff0c;为开发者带来全新视听盛宴 2024.9.19-9.21 云栖大会 即将上演“云原生AI”的全球盛会 展现最新的云计算技术发展与 AI技术融合之下的 “新探索” 一起来云栖小镇 见证3天的云原生AI 前沿探索…

828华为云征文 | 在华为云上通过Docker容器部署Elasticsearch并进行性能评测

目录 前言 1. 华为云X实例介绍及优势 1.1 柔性算力 1.2 vCPU和内存的灵活配比 1.3 成本效益与性能 2. 安装并运行 Docker 2.1 修改仓库配置文件 2.2 安装 Docker 2.3 启动 Docker 3. 使用Docker部署Elasticsearch 3.1 拉取Elasticsearch镜像 3.2 启动Elasticsearch…