0

0

c++中的友元函数和友元类_c++破坏封装的利器与陷阱

冰火之心

冰火之心

发布时间:2025-12-03 14:59:02

|

544人浏览过

|

来源于php中文网

原创

友元函数是用friend关键字声明的非成员函数,可访问类的私有和保护成员。例如printData能访问MyClass的私有数据data,虽提升灵活性但破坏封装性。

c++中的友元函数和友元类_c++破坏封装的利器与陷阱

在C++中,友元函数和友元类是一种特殊的机制,允许外部函数或类访问某个类的私有(private)和保护(protected)成员。这种设计初衷是为了提高灵活性,尤其在运算符重载、工具函数封装等场景下非常有用。但与此同时,它也打破了面向对象编程中“封装”的基本原则,使用不当容易带来维护性和安全性问题。

什么是友元函数

友元函数不是类的成员函数,但它被授予访问该类所有成员(包括私有和保护成员)的权限。通过在类内部使用 friend 关键字声明即可。

例如:

class MyClass {
private:
    int data;
public:
    MyClass(int d) : data(d) {}
    
    // 声明友元函数
    friend void printData(const MyClass& obj);
};

// 友元函数定义
void printData(const MyClass& obj) {
    std::cout << "Data: " << obj.data << std::endl;  // 可直接访问私有成员
}

这里,printData 虽然不是 MyClass 的成员,却能访问其私有成员 data。这在实现输入输出操作符重载时很常见,比如 operator 经常被设为友元。

立即学习C++免费学习笔记(深入)”;

什么是友元类

当一个类被声明为另一个类的友元时,它可以访问后者的所有私有和保护成员。这适用于两个类之间存在紧密协作关系的场景。

千问APP
千问APP

阿里最强大模型官方AI助手

下载
class SecretBox {
private:
    std::string secret;
public:
    SecretBox(const std::string& s) : secret(s) {}
    
    // 声明 FriendClass 为友元类
    friend class KeyHolder;
};

class KeyHolder {
public:
    void revealSecret(const SecretBox& box) {
        std::cout << "Secret is: " << box.secret << std::endl; // 合法
    }
};

KeyHolder 类可以自由访问 SecretBox 的私有数据,体现了一种高度信任的关系。但这也意味着两者耦合度极高,修改一个可能影响另一个。

友元打破封装带来的风险

封装是面向对象的核心原则之一,目的是隐藏实现细节,仅暴露必要的接口。而友元机制直接绕过了这一限制。

  • 一旦类的私有成员被友元访问,这些成员实际上就不再是真正“私有”的了,任何对这些成员的修改都必须同步通知所有友元函数或类。
  • 过度使用友元会导致类之间的依赖关系复杂化,降低代码可维护性。
  • 调试困难:当私有数据被意外修改时,排查范围不再局限于类内部,还需检查所有友元。
  • 破坏信息隐藏:本应由类自身管理的状态,可能被外部随意操控,增加出错概率。

合理使用建议

虽然友元有风险,但在某些情况下仍是必要且优雅的解决方案。

  • 用于运算符重载,特别是流输出 operator 和输入 operator>>,这是最广泛接受的用法。
  • 在实现容器与迭代器时,迭代器类常作为容器的友元,以高效访问内部结构。
  • 两个类逻辑上属于同一模块,且需要深度协作时,可谨慎使用友元类。
  • 避免将普通工具函数或无关类设为友元;优先考虑提供公共 getter/setter 接口,或重构设计。
  • 尽量缩小友元范围:如果只需要访问某个函数,就只声明该函数为友元,而不是整个类。

基本上就这些。友元不是洪水猛兽,而是双刃剑。掌握好使用的边界,才能发挥其便利而不陷入陷阱。不恰当地追求“完全私有”可能让设计僵化,但盲目使用友元也会让封装形同虚设。关键在于权衡与清晰的设计意图表达。

相关专题

更多
java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1465

2023.10.24

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

228

2024.02.23

php三元运算符用法
php三元运算符用法

本专题整合了php三元运算符相关教程,阅读专题下面的文章了解更多详细内容。

85

2025.10.17

go语言 面向对象
go语言 面向对象

本专题整合了go语言面向对象相关内容,阅读专题下面的文章了解更多详细内容。

56

2025.09.05

java面向对象
java面向对象

本专题整合了java面向对象相关内容,阅读专题下面的文章了解更多详细内容。

49

2025.11.27

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1023

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

66

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

442

2025.12.29

PHP WebSocket 实时通信开发
PHP WebSocket 实时通信开发

本专题系统讲解 PHP 在实时通信与长连接场景中的应用实践,涵盖 WebSocket 协议原理、服务端连接管理、消息推送机制、心跳检测、断线重连以及与前端的实时交互实现。通过聊天系统、实时通知等案例,帮助开发者掌握 使用 PHP 构建实时通信与推送服务的完整开发流程,适用于即时消息与高互动性应用场景。

3

2026.01.19

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 3.8万人学习

Pandas 教程
Pandas 教程

共15课时 | 0.9万人学习

ASP 教程
ASP 教程

共34课时 | 3.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号