C++多重继承,虚基类与友元

news2024/12/23 13:26:45

一.多重继承

就是一个类继承多个基类;

class <派生类名>:<派生方式1><基类名1>,```<派生方式n><基类名n>

class Derived:public:Base1,public:Base2

上述形式:基类之间由逗号隔开,且必须指明继承方式,否则默认为私有继承;

1.注意事项:

a.各种派生方式对于基类成员在派生类中的访问权限与单继承相同;

b.在使用多继承时,对基类成员的访问应无二义性

 2.二义性问题:

多重继承需要解决的主要问题:标识符不唯一造成的二义性,当继承的多个基类中有同名成员时就会造成二义性问题,这是不被允许的;

class Base1
{
public:
	int x;
	int a();
	int b();
};
class Base2
{
	int x;
	int a();
public:
	float b();
};
class Derived :Base1, Base2
{
	;
};

void d(Derived& e)
{
	e.x = 10;
	e.a();
	e.b();
}

int main()
{
	Derived ob;
	return 0;
}

像上述:Base1和Base2中都有相应的 x,a,b成员,此时就会造成二义性;

如何解决?

(1)使用域运算符“::”(范围解析运算符)

派生类的基类之间没有继承关系,同时又没有共同的基类(基类之间不是父与子的关系,也不是兄弟关系)则在引用同名成员时,可以在成员名前加上类名和域运算符来区别来自不同的基类的成员;

void d(Derived& e)
{
	e.Base1::x = 10;
	e.Base1::a();
	e.Base2::b();
}

使用域运算符,将d函数修改为这个形式就可以调用相应的函数,不再有二义性;非常直观

(2)使用同名覆盖的原则

在派生类中重新定义与基类中同名的成员(如果是成员函数,则参数表也要相同,参数不同则为重载)以屏蔽掉基类中的同名成员,在引用这些同名的成员时,引用的就是派生类中的成员;

class Base
{
public:
	int x;
	void show()
	{
		cout << "This is Base x=" << x << endl;
	}
};

class Derived :public Base
{
public:
	int x;
	void show()
	{
		cout << "This is Derived x=" << x << endl;
	}
};

int main()
{
	Derived ob;
	ob.x = 5;
	ob.show();
	ob.Base::x = 12;
	ob.Base::show();
	return 0;
}

此时直接调用类对象的成员就是派生类中的,不会产生歧义;

(3)使用虚基类(后面再提)

3.多重继承的构造函数和析构函数

在C++中声明多重继承构造函数的一般形式:

<派生类名>(参数总表)::基类名1(参数表)……基类名n(参数表n),对象成员名1(对象成员参数表1)……对象成员名n(对象成员参数表n)

多重继承构造函数和析构函数的执行顺序与单继承相同,但是需要强调的是,基类之间的执行顺序是严格按照声明从左到右的顺序来的,与他们在定义派生类构造函数中的次序无关;

class Base1
{
	int x1;
public:
	Base1(int y1)
	{
		x1 = y1;
		cout << "constructing Based,x1=" << x1 << endl;
	}
	~Base1()
	{
		cout << "destructing Base1" << endl;
	}
};

class Base2
{
	int x2;
public:
	Base2(int y2)
	{
		x2 = y2;
		cout << "constructing Base2.x2=" << x2 << endl;
	}
	~Base2()
	{
		cout << "destructing Base2" << endl;
	}
};

class Derived :public Base2, public Base1
{
private:
	Base1 ob1;
	Base2 ob2;
public:
	Derived(int x, int y, int z, int v) :Base1(x), Base2(y), ob1(z), ob2(v)
	{
		cout << "constructing Derived" << endl;
	}
};

int main()
{
	Derived ob(1, 2, 3, 4);
	return 0;
}

上述代码的运行结果为:

由此我们可以理解上面那句话: 基类之间的执行顺序是严格按照声明从左到右的顺序来的,与他们在定义派生类构造函数中的次序无关;

在上述程序中,先继承的Base2,那么先执行基类Base2,之后再执行Base1;

在定义对象的时候,先定义的为ob1,那么先执行Base1的初始化,再执行Base2的初始化

二.虚基类

虚基类是指被virtual修饰,用来解决基类中由于同名成员的问题而产生的二义性问题;

1.虚基类的声明

在派生类的声明过程中进行的:class <派生类名>: virtual<派生方式><基类名>

只对紧跟在virtual后的基类起作用;

class Base1:virtual public Base

声明虚基类之后,虚基类的成员在进一步派生过程中和派生类一起维护同一个内存

class Base
{
protected:
	int x;
public:
	Base()
	{
		x = 1;
	}
};

class Base1 :virtual public Base
{
public:
	Base1()
	{
		cout << "constructing Base1,x=" << x << endl;
	}
};

class Base2 :virtual public Base
{
public:
	Base2()
	{
		cout << "constructing Base2,x=" << x << endl;
	}
};

class Derived :public Base1, public Base2
{
public:
	Derived()
	{
		cout << "constructing Derived x=" << x << endl;
	}
};

int main()
{
	Derived obj;
	return 0;
}

由于把公共基类Base声明为类Base1和Base2的虚基类,所以由类Base1和类Base2派生的类Derived只有一个基类Base,在这里虚基类就可以避免二义性;按照顺序执行虚基类1,虚基类2的构造函数;(默认构造函数不需要调用,且这里都是没有参数的构造函数)

2.虚基类的构造函数和初始化

虚基类的初始化与一般的多继承的初始化在语法上是一样的,但构造函数的执行顺序不同;

(1)基类的构造函数;(2)虚基类的构造函数1;虚基类的构造函数2......(3)非虚基类的构造函数

class Base
{
protected:
	int x;
public:
	Base(int x1)
	{
		x = x1;
		cout << "constructing Base,x=" << x << endl;
	}
};

class Base1 :virtual public Base
{
	int y;
public:
	Base1(int x1, int y1) :Base(x1)
	{
		y = y1;
		cout << "constructing Base1,y=" << y << endl;
	}
};

class Base2 :virtual public Base
{
	int z;
public:
	Base2(int x1, int z1) :Base(x1)
	{
		z = z1;
		cout << "constructing Base2,z=" << z << endl;
	}
};

class Derived :public Base1, public Base2
{
	int xyz;
public:
	Derived(int x1, int y1, int z1, int xyz1) :Base(x1), Base1(x1, y1), Base2(x1, z1)
	{
		xyz = xyz1;
		cout << "constructing Derived xyz=" << xyz << endl;
	}
};

int main()
{
	Derived obj(1, 2, 3, 4);
	return 0;
}

 

在这里我们可以看出:虚基类Base的构造函数只执行了一次,因为当派生类Derived调用了虚基类Base的构造函数后,类Base1和Base2对虚基类的构造函数调用被忽略了!

注意事项:

a.virtual和派生方式的关键字的书写位置无关紧要,可以先写虚基类的关键字,也可以先写派生方式的关键字;

b.一个基类在作为某些类的虚基类的同时也可以作为另一些类的非虚基类;

c.虚基类构造函数的参数必须是由最新派生出来的类负责初始化,即使不是直接继承也是这样!(就是上面的derived:Base(x),初始化x)

三.友元

友元是用关键字friend修饰的,它提供了不同的类或对象的成员函数之间,类的成员函数与一般函数之间进行数据共享的机制

1.友元的引入

引入友元的目的:为了使类的私有成员和保护成员能够被其他类或其他成员函数访问

分类:(1)友元函数:友元是普通函数或类的成员函数;(2)友元类:友元是一个类

(友元类的所有成员函数都称为友元函数);

2.友元函数

友元函数不是当前类的成员函数,而是独立于当前类的外部函数它可以是普通函数或者其他类的成员函数。友元函数定义后可以访问该类的所有对象的所有成员函数,包括私有成员,保护成员和公有成员;

友元函数使用之前必须要在类定义时声明:其定义可以在类内部进行,也可以在类外部进行,但通常都定义在类的外部

C++将普通函数声明为友元函数的一般形式:

friend<数据类型><友元函数名>(参数表)

eg:friend double area(Rectangle &rectangle);

class Rectangle
{
	double length, width;
public:
	Rectangle(double a = 0, double b = 0)
	{
		length = a;
		width = b;
	}
	//Rectangle(Rectangle& r);//不重载也完全没影响啊!!!
	friend double area(Rectangle& rectangle);

};

double area(Rectangle& rectangle)
{
	return(rectangle.width * rectangle.length);
}
int main()
{
	Rectangle ob(4, 5);
	cout << "The area is:" << area(ob) << endl;
	return 0;
}

上述程序,如果把friend关键字去掉,那么在函数外定义的area函数就会报错,因为width和length都是类的私有成员,不可访问;但是加上友元定义就可以访问;

3.友元成员

如果一个类的成员函数是另一个类的友元函数,那么称这个成员函数为友元成员,那么通过该成员函数可以访问另一个类的成员,也可以访问自己所在类的成员;

——可以使得两个类相互访问;

class boy;
class girl
{
	char* name;
	int age;
public:
	girl(const char* n, int a)
	{
		name = new char[strlen(n) + 1];

		strcpy(name, n);
		age = a;
	}
	void prt(boy& b);
};

class boy
{
	char* name;
	int age;
public:
	boy(const char* n, int a)
	{
		name = new char[strlen(n) + 1];
		strcpy(name, n);
		age = a;
	}
    friend void girl::prt(boy& b);//声明友元成员
};

void girl::prt(boy& b)//定义友元成员
{
	cout << "girl's name" << name << " " << "age" << age << endl;
	cout << "boy's name" << b.name << " " << "age" << b.age << endl;
}

int main()
{
	girl gl("ztn", 19);
	boy bl("zhang", 18);
	gl.prt(bl);
	return 0;
}

在boy类内声明了girl类的成员函数prt为其友元,girl的prt既可以访问girl类的成员,也可以访问boy类的数据成员name和age;

上述:声明友元成员,定义友元成员等与上面的友元函数的区别就在于,此时存在两个类,该有友元成员可以访问另一个类的私有成员,保护成员和公有成员等;

上面也谈到:友元可以实现一般函数与成员函数之间,不同类之间进行数据共享的方式,那个人理解:友元函数:就可以实现一般函数与成员函数之间的数据共享,而友元成员则实现不同类之间的数据共享

使用友元成员需注意:

(1)必须先定义成员函数所在的类;上述例子中就是先定义了girl

(2)声明友元函数时,要加上成员函数所在类的类名和作用域,使用域运算符

(3)在主函数中一定要创建类girl的一个对象,只有这样才能通过对象名调用友元函数。

(4)如果在类定义前要使用到该类的成员,需要在使用前对该类进行声明,如class boy;(那么在上面的程序中,就是先声明友元函数所在的类,但是并不定义,先将友元成员所在的类进行声明定义,再定义友元函数的所在的类)

4.友元类

一个类作为另一个类的友元,称这个类为友元类。友元类中的所有成员函数都成为另一个类的友元函数。友元类中的所有成员函数都可以通过对象名直接访问另一个类中的所有成员从而实现不同类之间的数据共享

friend class<友元类名>——friend class girl

class boy;
class girl
{
	char* name;
	int age;
public:
	girl(const char* n, int a)
	{
		name = new char[strlen(n) + 1];
		strcpy(name, n);
		age = a;
	}
	void prt(boy& b);
};

class boy
{
	char* name;
	int age;
	friend class girl;
public:
	boy(const char* n, int a)
	{
		name = new char[strlen(n) + 1];
		strcpy(name, n);
		age = a;
	}
};

void girl::prt(boy& b)
{
	cout << "girl's name" << name << " " << "age" << age << endl;
	cout << "boy's name" << b.name << " " << "age" << b.age << endl;
}

int main()
{
	girl gl("ztn", 18);
	boy bl("zhang", 19);
	gl.prt(bl);
	return 0;
}

其实可以看出和友元成员类似,最终的运行结果也一样!都实现了不同类之间的数据共享。


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

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

相关文章

Redis-数据类型-String

文章目录 1、通过客户端连接redis2、查看当前数据库的key的数量3、切换数据库3.1、切换到第一个数据库3.2、切换到第二个数据库3.3、切换到默认的数据库&#xff0c;第0个数据库 4、当前数据库没有数据5、添加键值对6、查看当前库所有key7、清空当前库8、设置存活的秒数&#x…

定个小目标之刷LeetCode热题(25)

这道题采用的解法是桶排序&#xff0c;画草图如下 代码如下 //基于桶排序求解「前 K 个高频元素」 class Solution {public int[] topKFrequent(int[] nums, int k) {HashMap<Integer, Integer> map new HashMap();for (int num : nums) {if (map.containsKey(num)) {m…

python自动化系列:自动复制一个工作簿的所有工作表到其他工作簿

作品介绍 作品名称&#xff1a;自动复制一个工作簿的所有工作表到其他工作簿 开发环境&#xff1a;PyCharm 2023.3.4 python3.7 用到的库&#xff1a;os、xlwings 作品效果&#xff1a; 实现过程 一、代码设计 以下是代码的详细说明&#xff1a; 导入模块&#xff1a; …

物联网主机E6000:动环监控的全新解决方案!

物联网主机E6000在动环监控中的应用&#xff0c;标志着一场新的技术革命。随着科技的进步&#xff0c;特别是在物联网领域&#xff0c;数据采集和处理已经成为企业运营不可或缺的一环。 E6000作为一款支持多协议、多接口的全能型物联网主机&#xff0c;其在动环监控领域的应用…

详解Spring AOP(二)

目录 1.切点表达式 1.1execution表达式 1.2 annotation 1.2.1自定义注解MyAspect 1.2.3添加自定义注解 2.Sping AOP原理 2.1代理模式 2.1.1静态代理 2.1.2动态代理 2.1.3JDK动态代理 2.1.4CGLIB动态代理 3.总结 承接上文&#xff1a;详解Spring AOP&#xff08;一&…

【GO】rotatelogs库和sirupsen/logrus库实现日志功能的实践用例

“github.com/sirupsen/logrus” 是一个 Go 语言的日志库&#xff0c;它提供了一种简单、灵活的方式来记录日志。该库的主要特点包括&#xff1a; 支持多种日志输出目标&#xff0c;如控制台、文件等。 支持日志轮转&#xff0c;可以按照时间或文件大小进行轮转。 支持日志格式…

【投稿优惠|稳定出版】2024年体育、健康与食品安全国际学术会议(ICSHFS 2024)

【投稿优惠|稳定出版】2024年体育、健康与食品安全国际学术会议&#xff08;ICSHFS 2024&#xff09; 2024 International Conference on Sports, Health, and Food Safety(ICSHFS 2024) 会议简介&#xff1a; 2024年体育、健康与食品安全国际学术会议&#xff08;ICSHFS 2024…

Navicat和SQLynx功能比较三(数据导出:使用MySQL近千万数据测试)

数据导出的功能在数据库管理工具中是最普遍的功能之一。所以数据导出的功能稳定性和性能也是数据库管理工具是否能很好地满足应用需求的一个考虑因素。 目录 1. 整体比较 2. 示例 2.1 前置环境 2.2 Navicat导出 2.3 SQLynx导出 2.4 性能对比结果&#xff08;690万行数据&…

用友 打印模版增加打印次数,以付款申请单为例

一些公司需要在纸质单据上加上和电子发票一样的打印次数&#xff0c;具体做法如下&#xff1a; 找到要增加的单据&#xff0c;点击【格式设置】 找到打印&#xff0c;活动文本 设置活动文本&#xff0c;高级属性&#xff0c;在下拉框里找到【打印或显示操作员/次数/时间】或【…

NetSuite 审批工作流与事务处理类型的限制关系

在最近的实践中&#xff0c;用户提出可否对Credit Memo与Vendor Prepayment Application两种事务处理类型进行审批参与&#xff0c;当提出来的时候我们并没有直接在系统中进行测试&#xff0c;而是以常规事务处理的角度认为可以满足客户的需求&#xff1b; 但在沙盒环境中讨论…

【ARMv8/v9 GIC 系列 2.1 -- GIC SPI 中断的 pending 和 clear pending 配置】

文章目录 GIC Pending 和 Clear PendingGICD_ISPENDR<n>GICD_ICPENDR<n>参数<n>编号解释使用举例设置中断ID 100为挂起状态清除中断ID 100的挂起状态 代码实现小结 GIC Pending 和 Clear Pending 在ARMv8体系结构中&#xff0c;GICD_ISPENDR<n> 和 GI…

网页抓取和网页爬取之间有何区别?

随着互联网的发展和信息的爆炸式增长&#xff0c;数据收集和处理已成为企业和个人不可或缺的需求。在此背景下&#xff0c;网页抓取和网络爬虫已成为两种常见的数据收集方法。虽然这两种方法看似相似&#xff0c;但它们的方法和目标存在显著差异。本文将为您详细介绍网页抓取和…

看见未来社区:视频孪生技术打造智慧社区

智慧社区的建设需要创新的技术支撑。智汇云舟创新升级数字孪生为视频孪生技术&#xff0c;通过将真实世界的视频监控与数字模型实时融合&#xff0c;实现了对物理空间的实时实景动态模拟。 针对智慧社区管理业务&#xff0c;以智汇云舟视频孪生平台为支撑&#xff0c;综合承载…

从零开始搭建创业公司全新技术栈解决方案

从零开始搭建创业公司全新技术栈解决方案 关于猫头虎 大家好&#xff0c;我是猫头虎&#xff0c;别名猫头虎博主&#xff0c;擅长的技术领域包括云原生、前端、后端、运维和AI。我的博客主要分享技术教程、bug解决思路、开发工具教程、前沿科技资讯、产品评测图文、产品使用体…

python如何做报表系统

首先我们安装的python和PyQt5要保持一致&#xff0c;要么都是32位或者都是64位。 下载安装&#xff0c;安装完成之后我们记得要设置环境变量。 一路选择“下一步”就可以了。 安装完成之后我们需要验证是否成功。 pyqt5的安装直接安装就可以的&#xff0c;主要更改环境变量~~\p…

使用 Python 中的美丽汤进行网络数据解析的完整指南

Beautiful Soup 是一个广泛使用的 Python 库&#xff0c;在数据提取方面发挥着重要作用。它为解析 HTML 和 XML 文档提供了强大的工具&#xff0c;使从网页中轻松提取有价值的数据成为可能。该库简化了处理互联网上非结构化内容的复杂过程&#xff0c;使您可以将原始网页数据转…

房间灰尘多怎么办?资深保洁推荐除尘最有效的空气净化器

家中的灰尘问题一直是许多人的烦恼&#xff0c;尤其是对尘螨过敏的人来说&#xff0c;灰尘简直是“心头之患”。常言道&#xff1a;“家有尘埃&#xff0c;心头有累。”每天打扫灰尘成了许多人的烦恼&#xff0c;尤其是对尘螨过敏的人来说&#xff0c;灰尘简直是“心头之患”。…

健身器械行业外贸ERP管理降本增效解决方案

随着经济的迅速发展&#xff0c;以及健身锻炼的普及&#xff0c;人们对健身器材的需求量也在大幅度增加。欧美市场增长迅猛&#xff0c;家用健身器材热度飙升&#xff0c;尤其是跑步机、健身单车等轻便型家用健身器材&#xff0c;备受消费者青睐。 出口的主要国家包括&#xf…

主存储器的基本组成+容量扩展+与CPU的连接

1.基本组成 1.主存储器的基本组成和读写操作 主存储器被称为主存/内存。是计算机中存储程序的重要部件 主存储器内部包含了存储体、各种逻辑部件以及控制电路等。 主存是通过寻址的方式对存储体内的存储单元进行读写操作的。 主存首先要从MAR获取地址&#xff0c;之后译码器…

Paper Reading: EfficientAD:毫秒级延迟的准确视觉异常检测

EfficientAD 简介方法高效的patch描述PDN教师pretraining 轻量级的师生模型逻辑异常检测异常图像的标准化 实验局限性 EfficientAD: Accurate Visual Anomaly Detection at Millisecond-Level Latencies EfficientAD&#xff1a;毫秒级延迟的准确视觉异常检测, WACV 2024 paper…