C++中的‘friend‘关键字

目录

[友元函数(Friend Function)](#友元函数(Friend Function))

[友元类(Friend Class)](#友元类(Friend Class))

[友元函数模板(Friend Function Template)](#友元函数模板(Friend Function Template))

[友元类模板(Friend Class Template)](#友元类模板(Friend Class Template))

总结


摘要

在 C++ 中,`friend` 关键字用于声明一个函数或类为另一个类的友元,这样该函数或类可以访问这个类的私有成员和保护成员。友元机制提供了一种在保持数据封装的同时允许某些外部函数或类访问私有数据的方法。主要包含友元函数、友元类、友元函数模板、友元类模板。

友元函数(Friend Function)

友元函数是一个被指定为友元的非成员函数,它能够访问类的私有成员和保护成员。

声明和定义友元函数

cpp 复制代码
// `display` 函数被声明为 `MyClass` 的友元,因此它可以访问 `MyClass` 的私有成员 `value`
#include <iostream>

class MyClass {
public:
    MyClass(int val) : value(val) {}

    // 声明 friend 函数
    friend void display(const MyClass& obj);

private:
    int value;
};

// 定义 friend 函数
void display(const MyClass& obj) {
    std::cout << "Value: " << obj.value << std::endl;
}

int main() {
    MyClass obj(10);
    display(obj); // 输出: Value: 10
    return 0;
}

友元类(Friend Class)

友元类是一个被指定为友元的类,这样该类的所有成员函数都可以访问另一个类的私有成员和保护成员。

声明友元类

cpp 复制代码
// 类 `B` 被声明为类 `A` 的友元,因此 `B` 类中的成员函数 `showAValue` 可以访问 `A` 的私有成员 `value`
#include <iostream>

class B; // 前向声明

class A {
public:
    A(int val) : value(val) {}

    // 声明 B 为友元类
    friend class B;

private:
    int value;
};

class B {
public:
    void showAValue(const A& obj) {
        std::cout << "A's value: " << obj.value << std::endl;
    }
};

int main() {
    A a(20);
    B b;
    b.showAValue(a); // 输出: A's value: 20
    return 0;
}

友元函数模板(Friend Function Template)

我们可以将函数模板声明为友元,使得该模板实例化的每个函数都可以访问类的私有成员和保护成员。

声明友元函数模板

cpp 复制代码
// `showValue` 函数模板被声明为 `MyClass` 类模板的友元,因此它可以访问 `MyClass` 的私有成员 `value`
#include <iostream>

template <typename T>
class MyClass;

template <typename T>
void showValue(const MyClass<T>& obj);

template <typename T>
class MyClass {
public:
    MyClass(T val) : value(val) {}

    // 声明友元函数模板
    friend void showValue<>(const MyClass<T>& obj);

private:
    T value;
};

template <typename T>
void showValue(const MyClass<T>& obj) {
    std::cout << "Value: " << obj.value << std::endl;
}

int main() {
    MyClass<int> obj(30);
    showValue(obj); // 输出: Value: 30
    return 0;
}

友元类模板(Friend Class Template)

我们也可以将类模板声明为友元,使得该模板实例化的每个类都可以访问另一个类的私有成员和保护成员。

声明友元类模板

cpp 复制代码
// 类模板 `B` 被声明为类模板 `A` 的友元,因此 `B` 类模板的实例化对象可以访问 `A` 类模板的私有成员 `value`
#include <iostream>

template <typename T>
class B;

template <typename T>
class A {
public:
    A(T val) : value(val) {}

    // 声明友元类模板
    friend class B<T>;

private:
    T value;
};

template <typename T>
class B {
public:
    void showAValue(const A<T>& obj) {
        std::cout << "A's value: " << obj.value << std::endl;
    }
};

int main() {
    A<int> a(40);
    B<int> b;
    b.showAValue(a); // 输出: A's value: 40
    return 0;
}

总结

  1. 友元破坏封装:友元函数和友元类破坏了类的封装性,因为它们可以访问类的私有成员和保护成员,所以我们要谨慎使用友元机制。

  2. 友元关系单向性:友元关系是单向的。如果类 `A ` 是类 `B ` 的友元,类 `B ` 不能自动成为类 `A ` 的友元,除非也明确声明。

  3. 不能继承友元关系:友元关系不能被继承。如果基类将一个函数或类声明为友元,派生类不会自动继承这种友元关系。

  4. 滥用友元:避免滥用友元机制。如果大量使用友元,可能需要重新考虑类的设计,确保类的职责分明和接口的合理性。

友元机制在 C++ 中提供了一种在保持数据封装的同时允许某些外部函数或类访问私有数据的方法,应谨慎使用以避免破坏类的封装性和增加代码的复杂性。

相关推荐
山栀shanzhi9 分钟前
C/C++之:构造函数为什么不能设置为虚函数?
开发语言·c++·面试
lsx20240611 分钟前
.toggleClass() 方法详解
开发语言
谭欣辰16 分钟前
C++ 版Dijkstra 算法详解
c++·算法·图论
yuan1999717 分钟前
C&CG(列与约束生成)算法,来解决“风光随机性”下的微网鲁棒配置问题
c语言·开发语言·算法
李白的天不白26 分钟前
读到数据为undefind是的几种情况
开发语言·javascript·ecmascript
数智化精益手记局33 分钟前
人员排班管理软件的自动化功能解析:解决传统手工人员进行排班管理耗时长的难题
运维·数据结构·人工智能·信息可视化·自动化·制造·精益工程
wayz1138 分钟前
Day 11 编程实战:XGBoost金融预测与调参
算法·机器学习·金融·集成学习·boosting
念越39 分钟前
算法每日一题 Day07|双指针求解和为S的两个数
算法·力扣
LeocenaY41 分钟前
C语言面试题总结
c语言·开发语言·数据结构
qeen871 小时前
【算法笔记】双指针及其经典例题解析
c++·笔记·算法·双指针