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。
缺点:
破坏了封装性,可能导致类的耦合性提高。
友元是功能强大的工具,但应谨慎使用,以保持代码的可维护性。
热门推荐
物理专业学什么课程 学习难度大吗
36岁羽坛名将李宗伟患癌喉咙全烂,花近1000万续命,如今怎样了?
手机天翼云盘如何使用磁力链接下载文件?
住叠墅是一种什么样的体验?下叠、上叠和中叠,大有不同
莫桑石和钻石哪个更闪?
数字化口腔种植——缺牙患者的福音
为什么老年人更需要运动
超敏C反应蛋白88,身体炎症的信号?
常见的离子晶体结构介绍
本命年需要注意的事项
哇!原来口语可以这么溜!
哲学家论死亡与来世:灵魂不朽
武侠游戏——文化基因、情感共鸣与技术革新的三重奏
奢华与复古的酒吧,包裹一个浪漫的内在 | 酒吧设计
韩系眼影「眼头打亮」技巧!眼影打亮重点修饰眼型、调整轮廓,肿泡眼也能画出干净眼妆
人到中年吃什么最保健身体
2040铝型材单根承重能力详解:影响因素、计算方法及应用实践
如何选择符合国际标准的铝型材
面试中如何精准识别候选人匹配度?资深HR的5个评估维度
交通事故报警该按照怎么做
明日方舟群奶角色深度解析:高效资源管理策略
泰国电商跨境交易规则全解析及企业应对策略
龙虾养殖需要什么条件
NVIDIA GT720 与 GT420 显卡性能对比及适用场景分析
种植牙后能否食用鸽子肉?了解种植牙与饮食的关系
种植牙后应该注意哪些饮食事项?如何选择适合的食物以促进恢复?
破圈3000万!《燕云十六声》如何用武侠文化重构游戏叙事?
居家装修玻璃选购全攻略:从性能到安装注意事项
两个人分手前,会有这些征兆
如何判断面试者是真牛,还是吹牛?