c++ 学习 之 友元

前言

C++ 中的友元(Friend)是一种特殊的关系,允许一个类或函数访问另一个类的私有成员。友元在某些情况下非常有用,但也应该谨慎使用,因为它会破坏封装性,增加代码的耦合性。

正文

全局函数做友元

我们先看看下面的代码

#define CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;

class Building
{
public:
	Building()
	{
		m_SittingRoom = "客厅";
		m_Bedroom = '卧室';
	}
public:
	string m_SittingRoom;

private:
	string m_Bedroom;
};


// 全局函数

void goodGay(Building* building)
{
	cout << "好基友全局函数正在访问 : " << building->m_SittingRoom << endl;

	cout << "好基友全局函数正在访问 : " << building->m_Bedroom << endl;
}


void test()
{
	Building b;
	goodGay(&b);
}
int main()
{
	
	test();
}

上面代码中会出错,因为 m_Bedroom 属性是 private 的 ,外部无法访问

下面看看如何解决

#define CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;

class Building
{
	// 可以理解为 goodGay 是 Building 好朋友,可以访问 Building 中私有成员
	friend void goodGay(Building* building);
public:
	Building()
	{
		m_SittingRoom = "客厅";
		m_Bedroom = "卧室";
	}
public:
	string m_SittingRoom;

private:
	string m_Bedroom;
};


// 全局函数

void goodGay(Building* building)
{
	cout << "好基友全局函数正在访问 : " << building->m_SittingRoom << endl;

	cout << "好基友全局函数正在访问 : " << building->m_Bedroom << endl;

}


void test()
{
	Building b;
	goodGay(&b);
}
int main()
{
	
	test();
}

类做友元

看看下面的代码

#define CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;

class Building {

	
public:
	Building();
	string m_SittingRoom; // 客厅

private:

	string m_Bedroom; //卧室
};

Building::Building()
{
	m_SittingRoom = "客厅";
	m_Bedroom = "卧室";
}

class GoodGay {

	
public:
	GoodGay();
	void visit();
	Building* building;
};

GoodGay::GoodGay()
{

	// 创建 建筑物对象
	building = new Building;
}

void GoodGay::visit()
{
	cout << " 好基友类正在 访问" << building->m_SittingRoom << endl;

	cout << " 好基友类正在 访问" << building->m_Bedroom << endl;
}

void test()
{
	GoodGay g;
	g.visit();
}

int main()
{
	test();
	
}

也会出现无法访问 私有变量 m_BedRoom 的报错
怎么解决呢

#define CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;

class Building {
	// GoodGay 类 是这个类的好朋友,可以访问私有类中的好朋友
	friend class GoodGay;
	
public:
	Building();
	string m_SittingRoom; // 客厅

private:

	string m_Bedroom; //卧室
};

Building::Building()
{
	m_SittingRoom = "客厅";
	m_Bedroom = "卧室";
}

class GoodGay {

	
public:
	GoodGay();
	void visit();
	Building* building;
};

GoodGay::GoodGay()
{

	// 创建 建筑物对象
	building = new Building;
}

void GoodGay::visit()
{
	cout << " 好基友类正在 访问" << building->m_SittingRoom << endl;

	cout << " 好基友类正在 访问" << building->m_Bedroom << endl;
}

void test()
{
	GoodGay g;
	g.visit();
}

int main()
{
	test();
	
}

成员函数做友元

看看下面的代码

#define CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;

class Building
{

public:
	Building();
public:
	string m_Sittingroom;

private:
	string m_Bedroom;

};

Building::Building()
{
	m_Sittingroom = "客厅";
	m_Bedroom = "卧室";
}

class GoodGay
{
public:
	GoodGay();

	void visit();  // 让这个函数可以访问 Building 中私有成员

	void visit02();  // 这个函数不可以访问 Building 中的私有成员

	Building* building;
};

GoodGay::GoodGay()
{
	building = new Building;
}

void GoodGay::visit()
{
	cout << " visit 函数 正在访问 " << building->m_Sittingroom << endl;

	cout << " visit 函数 正在访问 " << building->m_Bedroom << endl;
}

void GoodGay::visit02()
{
	cout << " visit02 函数 正在访问 " << building->m_Sittingroom << endl;

	cout << " visit02 函数 正在访问 " << building->m_Bedroom << endl;
}

void test()
{
	GoodGay g;
	g.visit();
	g.visit02();
}

int main()
{
	test();
	
}

还是老问题,会报错
看看怎么解决呢

#define CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;


class Building;
class GoodGay
{

public:
	GoodGay();

	void visit();  // 让这个函数可以访问 Building 中私有成员

	void visit02();  // 这个函数不可以访问 Building 中的私有成员

	Building* building;
};
class Building
{
	friend void GoodGay:: visit();
public:
	Building();
public:
	string m_Sittingroom;

private:
	string m_Bedroom;

};

Building::Building()
{
	m_Sittingroom = "客厅";
	m_Bedroom = "卧室";
}



GoodGay::GoodGay()
{
	building = new Building;
}

void GoodGay::visit()
{
	cout << " visit 函数 正在访问 " << building->m_Sittingroom << endl;

	cout << " visit 函数 正在访问 " << building->m_Bedroom << endl;
}

void GoodGay::visit02()
{
	cout << " visit02 函数 正在访问 " << building->m_Sittingroom << endl;

	//cout << " visit02 函数 正在访问 " << building->m_Bedroom << endl;
}

void test()
{
	GoodGay g;
	g.visit();
	g.visit02();
}

int main()
{
	test();

}

深入学习

class Building;
这行代码到底需不需要
class Building; 是一个前向声明(forward declaration),它告诉编译器关于 Building 类的存在,但不提供有关该类的详细信息。这是因为在 GoodGay 类中使用了一个指向 Building 类的指针成员 building,而在这个地方编译器需要知道 Building 类的存在以正确编译代码。

如果没有 class Building; 这行代码,编译器将无法识别 Building 类,并且会报错,因为在 GoodGay 类中使用了 Building 类的指针,而编译器不知道 Building 类的定义,因此无法正确解析 GoodGay 类的定义。

通过添加 class Building; 这行代码,你告诉编译器有一个名为 Building 的类存在,即使在后面的代码中还没有提供该类的详细定义。这使得编译器能够在 GoodGay 类中正确识别 building 成员的类型,并继续编译代码。然后,当编译器在后面找到 Building 类的实际定义时,它将确保 GoodGay 类中的使用是合法的。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

wniuniu_

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值