当前位置: 首页 > article >正文

C++12--友元

友元提供了一种突破封装的方式,有时提供了便利。但是友元会增加耦合度,破坏了封装,所以友元不宜多用。

1.友元函数

问题:现在尝试去重载operator<<,然后发现没办法将operator<<重载成成员函数。因为cout的输出流对象和隐含的this指针在占用第一个参数的位置。this指针默认时第一个参数也就是左操作数了。但是实际使用中cout需要是第一个形参对象,才能正常使用。所以要将operator<<重载成全局函数。但又会导致类外没办法访问成员,此时就需要友元来解决。

class Date
{
public:
	Date(int year, int month, int day)
		:_year(year)
		,_month(_month)
		,_day(day)
	{}

	//d1<<cout;->d1.operator<<(&d1,cout);不符合常规调用
	//因为成员函数第一个参数一定是隐藏的this,所以d1必须放在<<的左侧
	ostream& operator<<(ostream& _cout)
	{
		_cout << _year << "-" << _month << "-" << _day << endl;
		return _cout;
	}

private:
	int _year;
	int _month;
	int _day;
};

友元函数可以直接访问类的私有成员,它是定义在类外部的普通函数,不属于任何类,但需要在类的内部声明,声明时需要加friend关键字。

class Date
{
	friend	ostream& operator<<(ostream& _cout, const Date& d);
	friend istream& operator>>(istream& _cin, Date& d);

public:
	Date(int year = 2024, int month = 12, int day = 19)
		:_year(year)
		,_month(_month)
		,_day(day)
	{}

	//d1<<cout;->d1.operator<<(&d1,cout);不符合常规调用
	//因为成员函数第一个参数一定是隐藏的this,所以d1必须放在<<的左侧
	ostream& operator<<(ostream& _cout)
	{
		_cout << _year << "-" << _month << "-" << _day << endl;
		return _cout;
	}

private:
	int _year;
	int _month;
	int _day;
};

ostream& operator<<(ostream& _cout, const Date& d)
{
	_cout << d._year << "-" << d._month << "-" << d._day;
	return _cout;
}

istream& operator>>(istream& _cin, Date& d)
{
	_cin >> d._year;
	_cin >> d._month;
	_cin >> d._day;
	return _cin;
}

int main()
{
	Date d;
	cin >> d;
	cout << d << endl;
	return 0;
}

说明:

  • 友元函数可访问类的私有和保护成员,但不是类的成员函数
  • 友元函数不能用const修饰
  • 友元函数可以在类定义的任何地方声明,不受类访问限定符限制
  • 一个函数可以是多个类的友元函数
  • 友元函数的调用与普通函数的调用原理相同

2.友元类

友元类的所有成员函数都可以是另一个类的友元函数,都可以访问另一个类中的非公有成员。

  • 友元关系是单向的,不具有交换性。
  • 友元关系不能传递(如果B是A的友元,C是B的友元,则不能说明C是A的友元)
  • 友元关系不能继承
class Time
{
	friend class Date;//声明日期类为时间类的友元,则在日期类中就直接访问Time类中的私有或
	                  //成员变量

public:
	Time(int hour = 0, int minute = 0, int second = 0)
		:_hour(hour)
		,_minute(minute)
		,_second(second)
	{}

private:
	int _hour;
	int _minute;
	int _second;
};

class Date
{
public:
	Date(int year = 2024, int month = 12, int day = 1)
		:_year(year)
		,_month(month)
		,_day(day)
	{}
	void SetTimeOfDate(int hour,int minute,int second)
	{
		//直接访问时间类私有的成员变量
		_t._hour = hour;
		_t._minute = minute;
		_t._second = second;
	}
private:
	int _year;
	int _month;
	int _day;
	Time _t;
};


http://www.kler.cn/a/444364.html

相关文章:

  • 为何VisualRules更适合技术人员使用
  • netcore 集成Prometheus
  • Unity3D仿星露谷物语开发7之事件创建动画
  • JumpServer开源堡垒机搭建及使用
  • Ansible 批量管理华为 CE 交换机
  • 个人秋招总结
  • 已有 containerd 的情况下部署二进制 docker 共存
  • 批量DWG文件转dxf(CAD图转dxf)——c#插件实现
  • (C语言)双向链表
  • iClent3D for Cesium 实现无人机巡检飞行效果
  • neo4j 图表数据导入到 TuGraph
  • 使用Docker启用MySQL8.0.11
  • HTMLCSS:这个动态删除按钮打几分?
  • 基于JavaScript的DBUtils增删改查操作实验
  • 算法编程题-不相交的线 联通网络的操作次数 销售价值减少的颜色球
  • git废弃指定文件的修改
  • Java设计模式 —— 【结构型模式】适配器模式(类的适配器、对象适配器、接口适配器)详解
  • 可视化平台FineReport的安装及简单使用
  • Windows 配置 Tomcat环境
  • 专业125+总分400+南京理工大学818考研经验南理工电子信息与通信工程,真题,大纲,参考书。
  • Python中map函数返回值类型用法介绍
  • arcgisPro将面要素转成CAD多段线
  • K8s HPA的常用功能介绍
  • 利用系统自带的存储感知功能清理系统中的升级补丁
  • Linux 定时任务操作详解及python简单的任务管理器
  • 设计模式-读书笔记2