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

相关推荐
摇滚侠2 分钟前
java http body的格式 ‌application/x-www-form-urlencoded‌不支持文件上传
java·开发语言·http
檀越剑指大厂12 分钟前
【Python系列】 Base64 编码:使用`base64`模块
开发语言·python
誓约酱16 分钟前
(动画)Qt控件 QLCDNumer
开发语言·c++·git·qt·编辑器
夫琅禾费米线16 分钟前
leetcode2650. 设计可取消函数 generator和Promise
开发语言·javascript·leetcode·ecmascript
Sunyanhui117 分钟前
力扣 三数之和-15
数据结构·算法·leetcode
Mr.kanglong17 分钟前
【LeetCode热题100】队列+宽搜
算法·leetcode·职场和发展
sp_fyf_202418 分钟前
计算机前沿技术-人工智能算法-大语言模型-最新研究进展-2024-11-05
人工智能·深度学习·神经网络·算法·机器学习·语言模型·数据挖掘
@小博的博客27 分钟前
C++初阶学习第十三弹——容器适配器和优先级队列的概念
开发语言·数据结构·c++·学习
钰见梵星30 分钟前
深度学习优化算法
人工智能·深度学习·算法
Dola_Pan30 分钟前
C语言:函数指针精讲
c语言·开发语言