C++中类的【友元】详解
创作时间:
作者:
@小白创作中心
C++中类的【友元】详解
引用
CSDN
1.
https://blog.csdn.net/chenai886/article/details/144667981
在C++中,友元是一种特殊的关系,允许某个函数或类访问另一个类的私有成员和保护成员。友元打破了类的封装性,但也为实现一些复杂逻辑提供了便利。
下面,我会用详细解释和大量代码实例帮助你通俗易懂地理解友元函数、友元类、友元模板的概念和实际用途。
一、友元函数
友元函数是一种非成员函数,但它可以直接访问一个类的私有和保护成员。它是通过
friend
关键字声明的。
友元函数的特点:
- 友元函数不是类的成员,但可以访问类的私有和保护成员。
- 友元函数需要在类中声明,定义可以在类外。
- 友元关系是单向的,类给函数友元权限,但函数没有反向权限。
1.1 单个友元函数
我们通过一个简单的例子说明友元函数的工作原理:
代码实例:访问私有成员
#include <iostream>
using namespace std;
class Box {
private:
double length; // 私有成员变量
public:
Box(double l) : length(l) {}
// 声明友元函数
friend double getLength(const Box& b);
};
// 友元函数的定义
double getLength(const Box& b) {
return b.length; // 可以直接访问 Box 的私有成员
}
int main() {
Box box(10.5);
cout << "Length of the box: " << getLength(box) << endl;
return 0;
}
输出:
Length of the box: 10.5
解释:
- getLength
是
Box
的友元函数。 - getLength
虽然不是
Box
的成员,但由于被声明为友元,可以直接访问
length
。
1.2 友元函数访问多个对象的私有成员
友元函数常用于操作两个或多个对象的私有成员。
代码实例:比较两个对象
#include <iostream>
using namespace std;
class Box {
private:
double length;
public:
Box(double l) : length(l) {}
// 声明友元函数
friend bool compare(const Box& b1, const Box& b2);
};
// 友元函数定义
bool compare(const Box& b1, const Box& b2) {
return b1.length > b2.length; // 可以访问 b1 和 b2 的私有成员
}
int main() {
Box box1(10.5), box2(15.0);
if (compare(box1, box2)) {
cout << "Box1 is larger" << endl;
} else {
cout << "Box2 is larger" << endl;
}
return 0;
}
输出:
Box2 is larger
二、友元类
友元类允许一个类的所有成员函数访问另一个类的私有和保护成员。
友元类的特点:
- 友元类声明后,该类的所有成员函数都能访问另一个类的私有和保护成员。
- 友元关系是单向的,声明友元的类允许被访问,但反之不成立。
2.1 友元类的简单实例
代码实例:一个类操作另一个类
#include <iostream>
using namespace std;
class Box; // 前向声明
class Calculator {
public:
double calculateVolume(const Box& b); // 计算体积
};
class Box {
private:
double length, width, height;
public:
Box(double l, double w, double h) : length(l), width(w), height(h) {}
// 声明友元类
friend class Calculator;
};
// 友元类成员函数的定义
double Calculator::calculateVolume(const Box& b) {
return b.length * b.width * b.height; // 访问 Box 的私有成员
}
int main() {
Box box(3.0, 4.0, 5.0);
Calculator calc;
cout << "Volume of the box: " << calc.calculateVolume(box) << endl;
return 0;
}
输出:
Volume of the box: 60
解释:
- Calculator
是
Box
的友元类。 - Calculator
的成员函数
calculateVolume
可以直接访问
Box
的私有成员。
2.2 友元类的双向访问
如果两个类需要相互访问对方的私有成员,则需要在每个类中分别声明友元。
代码实例:双向友元
#include <iostream>
using namespace std;
class B; // 前向声明
class A {
private:
int valueA;
public:
A(int val) : valueA(val) {}
// 声明 B 的成员函数为友元
friend void display(const A& a, const B& b);
};
class B {
private:
int valueB;
public:
B(int val) : valueB(val) {}
// 声明 A 的成员函数为友元
friend void display(const A& a, const B& b);
};
// 友元函数定义
void display(const A& a, const B& b) {
cout << "A: " << a.valueA << ", B: " << b.valueB << endl;
}
int main() {
A objA(10);
B objB(20);
display(objA, objB);
return 0;
}
输出:
A: 10, B: 20
解释:
- A
的私有成员
valueA
和
B
的私有成员
valueB
都可以在
display
函数中访问。 - 这需要双向声明友元关系。
三、友元模板
友元模板允许模板类或模板函数访问其他类的私有成员。
3.1 模板函数作为友元
代码实例:模板函数访问私有成员
#include <iostream>
using namespace std;
class Box {
private:
double length;
public:
Box(double l) : length(l) {}
// 声明模板函数为友元
template <typename T>
friend void displayLength(const T& obj);
};
template <typename T>
void displayLength(const T& obj) {
cout << "Length: " << obj.length << endl;
}
int main() {
Box box(10.5);
displayLength(box);
return 0;
}
输出:
Length: 10.5
3.2 模板类作为友元
代码实例:模板类访问私有成员
#include <iostream>
using namespace std;
template <typename T>
class Calculator;
class Box {
private:
double length;
public:
Box(double l) : length(l) {}
// 声明模板类为友元
template <typename T>
friend class Calculator;
};
template <typename T>
class Calculator {
public:
void printLength(const Box& b) {
cout << "Length: " << b.length << endl; // 访问私有成员
}
};
int main() {
Box box(10.5);
Calculator<int> calc;
calc.printLength(box);
return 0;
}
输出:
Length: 10.5
总结
友元的使用场景:
- 当需要跨类访问私有成员时(如运算符重载、对象比较)。
- 当需要将工具类设计为另一个类的友元(如计算器类访问目标类)。
- 需要实现灵活的模板操作。
优缺点:
优点:
提供了跨类访问的便利性。
实现复杂逻辑时,避免冗长的 getter/setter。
缺点:
破坏了封装性,可能导致类的耦合性提高。
友元是功能强大的工具,但应谨慎使用,以保持代码的可维护性。
热门推荐
查账征收个体工商户必看:申报个人所得税经营所得常见误区解析
X光一年只能拍一次?核磁有辐射?在医院拍这些“片子”有啥区别?
《道德经》第七十七章:天道与人道的对比
三分线外的艺术:精准投篮的科学训练法
胸椎骨折伤残鉴定标准及流程详解
“作业熔断”,减负良策还是焦虑催化剂?
PLC基础知识:15个核心知识点详解
PLC编程入门,掌握PLC基础知识与工作原理
鸡蛋可以放入微波炉吗?如何安全使用
香港优才申请:英语水平是否有强制要求?
在职女性怀孕期间享有哪些权益保障?
融合多模态与模糊推理的可解释性机器学习股价预测模型研究【附数据】
大牙烂了只剩一点空壳了怎么办
空运包装老踩雷?液体粉状货物包装要求
无人机动力系统测试:实测数据与CFD模拟仿真数据关联对比分析
项目文控管理是做什么的
文控文员的工作计划怎么写
夫妻共同买房全攻略:从贷款到产权登记的关键注意事项
氨基酸是什么 氨基酸的性质
猪心汤好喝有技巧,汤鲜味美不腥不柴,做法简单,熬夜失眠要常喝
儿童书桌高度多少合适?这份实用指南请收好
推荐使用法律数据库:了解法律法规的便捷途径
法律文书有效性查询的实务操作与法律依据
阴历与阳历的区别及其在生活中的重要性探讨
8个最佳开源内网穿透工具推荐
与痴呆患者、老年人的沟通技巧,五个原则请收好
惠州供电:创新“三线整治 四网合一”模式,助力“百千万工程”
色弱恢复最有效的方法
如何算闰年 闰年的由来
矩阵的基本概念