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

C++中类的【友元】详解

在C++中,友元是一种特殊的关系,允许某个函数或类访问另一个类的私有成员和保护成员。友元打破了类的封装性,但也为实现一些复杂逻辑提供了便利。

在这里插入图片描述

下面,我会用详细解释和大量代码实例帮助你通俗易懂地理解友元函数、友元类、友元模板的概念和实际用途。


一、友元函数

友元函数是一种非成员函数,但它可以直接访问一个类的私有和保护成员。它是通过 friend 关键字声明的。

友元函数的特点

  1. 友元函数不是类的成员,但可以访问类的私有和保护成员。
  2. 友元函数需要在类中声明,定义可以在类外。
  3. 友元关系是单向的,类给函数友元权限,但函数没有反向权限。

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
解释
  1. getLengthBox 的友元函数。
  2. 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

二、友元类

友元类允许一个类的所有成员函数访问另一个类的私有和保护成员。

友元类的特点

  1. 友元类声明后,该类的所有成员函数都能访问另一个类的私有和保护成员。
  2. 友元关系是单向的,声明友元的类允许被访问,但反之不成立。

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
解释
  1. CalculatorBox 的友元类。
  2. 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
解释
  1. A 的私有成员 valueAB 的私有成员 valueB 都可以在 display 函数中访问。
  2. 这需要双向声明友元关系。

三、友元模板

友元模板允许模板类或模板函数访问其他类的私有成员。

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

总结

友元的使用场景

  1. 当需要跨类访问私有成员时(如运算符重载、对象比较)。
  2. 当需要将工具类设计为另一个类的友元(如计算器类访问目标类)。
  3. 需要实现灵活的模板操作。

优缺点

  • 优点
    • 提供了跨类访问的便利性。
    • 实现复杂逻辑时,避免冗长的 getter/setter。
  • 缺点
    • 破坏了封装性,可能导致类的耦合性提高。

友元是功能强大的工具,但应谨慎使用,以保持代码的可维护性。


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

相关文章:

  • GitPuk安装配置指南
  • 使用 perf 工具进行性能分析
  • dify的ChatFlow自定义上传图片并通过HTTP请求到SpringBoot后端
  • 【ROS2】坐标TF变换工具-tf2_ros
  • 安装CPU版的torch(清华源)
  • K8S 黑魔法之如何从 Pod 拿到节点的命令行
  • 每天40分玩转Django:实操图片分享社区
  • css 编写注意-1-命名约定
  • selenium执行js
  • 2024年12月CCF-GESP编程能力等级认证Python编程四级真题解析
  • sqoop抽数报错Every derived table must have its own alias
  • 最长正则括号序列算法详解
  • ElementUI 的 form 表单校验
  • 深度学习——神经网络中前向传播、反向传播与梯度计算原理
  • 计算机网络B重修班-期末复习
  • 《探索PyTorch计算机视觉:原理、应用与实践》
  • 【数据可视化案列】白葡萄酒质量数据的EDA可视化分析
  • uniapp实现获取用户定位信息、手机号信息、蓝牙、设备、相册、相机、声音等,请你完善展示所有信息
  • 用VBA将word文档处理成支持弹出式注释的epub文档可用的html内容
  • Docker Compose 安装 Harbor
  • vue3标签中的ref属性如何使用$refs获取元素
  • postman关联接口用于登录(验证码会变情况)
  • QT:QLabel的LED透明跑马灯
  • 《信管通低代码信息管理系统开发平台》Linux环境安装说明
  • es 3期 第18节-分页查询使用避坑的一些事
  • UML 建模实验