0

0

C++中friend友元怎么理解_C++友元函数与友元类应用场景【深度】

尼克

尼克

发布时间:2026-02-06 14:20:39

|

369人浏览过

|

来源于php中文网

原创

友元函数能访问类的私有成员是因为编译器在编译期为其特许访问权限,不进行访问控制检查;它非类成员、无this指针、需显式传参;友元关系不可继承和传递,声明须在类内;常用于左操作数为非类类型的运算符重载。

c++中friend友元怎么理解_c++友元函数与友元类应用场景【深度】

友元函数为什么能访问类的私有成员

因为编译器在编译阶段就为友元函数“开了后门”——它不经过访问控制检查,直接生成对私有成员的符号引用。这不是运行时绕过封装,而是编译期特许:只要声明为 friend,该函数就获得和类内成员函数同等的访问权限。

常见错误是以为友元能“继承”或“传递”:比如把一个友元函数再传给另一个类作友元,这是无效的。友元关系不具有传递性,也不被继承。

  • friend 声明必须出现在类定义内部,且位置无关(public/protected/private 区域里都可)
  • 友元函数本身不是类的成员,没有 this 指针,所有参数都需显式传入
  • 若友元函数定义在类外,需提前声明(尤其涉及模板或跨文件时),否则链接时报 undefined reference

什么时候必须用友元函数而不是成员函数

典型场景是操作符重载中需要左操作数为非类类型,比如 operator 或 operator+ 的左值是 std::ostream 或内置类型时。

例如实现 std::cout ,operator 的第一个参数是 std::ostream&,不可能是 MyClass 的成员函数(成员函数隐含 this 为第一参数)。此时只能用非成员函数,并声明为友元才能读取 obj 的私有数据。

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

  • 流插入/提取运算符:operator、operator>>
  • 对称二元运算符:如 operator+ (int, MyClass)operator== (MyClass, MyClass)(当希望左右操作数类型顺序可互换)
  • 某些工厂函数或序列化辅助函数,需深度访问但又不适合暴露为 public 接口

友元类的典型误用与合理边界

友元类常被当成“内部模块解耦”的捷径,结果导致封装形同虚设。真正合理的使用场景极少:仅限于两个类构成逻辑上不可分的“一体两面”,且其中一方纯粹作为另一方的底层支撑存在。

腾讯智影
腾讯智影

腾讯推出的在线智能视频创作平台

下载

例如 std::string 和它的私有迭代器类(实际标准库不这样实现,但教学模型中常见),或某些容器与其配套的 const_iterator 类——后者只读访问容器私有数据,不提供修改能力,且生命周期完全受控于容器。

  • 友元类的所有成员函数(包括后续新增的)自动获得对目标类私有成员的完全访问权,无法按需限制
  • 若只是需要部分访问,优先考虑提供受限的 public 成员函数(如 data()size()),而非直接开友元
  • 跨模块使用友元类极易引发头文件依赖爆炸,尤其当友元类定义在另一个 translation unit 中时,需确保其完整定义在友元声明前可见

友元带来的 ABI 和维护风险

友元会把函数或类绑定到目标类的私有实现细节上。一旦私有成员名、类型或布局变化,所有友元代码都可能悄无声息地编译通过但行为异常——因为它们直接读写内存偏移,而编译器不会校验这种访问是否仍合法。

比方说,你给类加了一个新私有成员在前面,导致原有私有成员地址偏移改变,而友元函数仍在旧偏移处读取,结果拿到的是垃圾值。这种 bug 极难调试。

  • 单元测试很难覆盖友元路径,因为它绕过了公共接口契约
  • 发布 DLL 或 so 时,友元函数若在动态库外定义,会破坏二进制兼容性(因为私有布局不属于 ABI 承诺)
  • 现代 C++ 倾向用 key 类技巧(如 class secret_key { friend class A; secret_key() = default; };)替代粗粒度友元,把访问权限收缩到单个函数级别

友元不是封装的例外,而是封装边界的主动延展;延展多少,就得多承担多少耦合代价。真正棘手的不是语法怎么写,而是决定“谁值得被信任到能碰我的私有内存”。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

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

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

1510

2023.10.24

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

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

234

2024.02.23

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

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

127

2025.10.17

Go语言实现运算符重载有哪些方法
Go语言实现运算符重载有哪些方法

Go语言不支持运算符重载,但可以通过一些方法来模拟运算符重载的效果。使用函数重载来模拟运算符重载,可以为不同的类型定义不同的函数,以实现类似运算符重载的效果,通过函数重载,可以为不同的类型实现不同的操作。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

195

2024.02.23

undefined是什么
undefined是什么

undefined是代表一个值或变量不存在或未定义的状态。它可以作为默认值来判断一个变量是否已经被赋值,也可以用于设置默认参数值。尽管在不同的编程语言中,undefined可能具有不同的含义和用法,但理解undefined的概念可以帮助我们更好地理解和编写程序。本专题为大家提供undefined相关的各种文章、以及下载和课程。

5503

2023.07.31

网页undefined是什么意思
网页undefined是什么意思

网页undefined是指页面出现了未知错误的意思,提示undefined一般是在开发网站的时候定义不正确或是转换不正确,或是找不到定义才会提示undefined未定义这个错误。想了解更多的相关内容,可以阅读本专题下面的文章。

3126

2024.08.14

网页undefined啥意思
网页undefined啥意思

本专题整合了undefined相关内容,阅读下面的文章了解更多详细内容。后续继续更新。

875

2025.12.25

1688阿里巴巴货源平台入口与批发采购指南
1688阿里巴巴货源平台入口与批发采购指南

本专题整理了1688阿里巴巴批发进货平台的最新入口地址与在线采购指南,帮助用户快速找到官方网站入口,了解如何进行批发采购、货源选择以及厂家直销等功能,提升采购效率与平台使用体验。

21

2026.02.06

快手网页版入口与电脑端使用指南 快手官方短视频观看入口
快手网页版入口与电脑端使用指南 快手官方短视频观看入口

本专题汇总了快手网页版的最新入口地址和电脑版使用方法,详细提供快手官网直接访问链接、网页端操作教程,以及如何无需下载安装直接观看短视频的方式,帮助用户轻松浏览和观看快手短视频内容。

0

2026.02.06

热门下载

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

精品课程

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

共94课时 | 8.7万人学习

C 教程
C 教程

共75课时 | 4.5万人学习

C++教程
C++教程

共115课时 | 16.1万人学习

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

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