news 2026/4/20 8:48:06

《你真的了解C++吗》No.028:友元(friend)的必要性与边界——为什么它不是对封装的破坏?

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
《你真的了解C++吗》No.028:友元(friend)的必要性与边界——为什么它不是对封装的破坏?

《你真的了解C++吗》No.028:友元(friend)的必要性与边界——为什么它不是对封装的破坏?

导言:封装的“后门”还是“窗户”?

如果一个类将所有成员都设为private,它确实非常安全,但也可能变得非常孤立。有些操作(如操作符重载或跨类协作)必须访问内部数据才能高效运行。

友元机制允许一个类授予特定的函数或类“通行证”,让它们能够访问自己的非公有成员。


一、 友元存在的物理理由:操作符重载

为什么我们需要友元?最经典的例子是operator<<的重载。

如果你想让你的类支持std::cout << myObj;,你不能把它写成成员函数。因为成员函数的第一个参数必须是类对象本身(this),这意味着你只能写成myObj << std::cout;,这违背了直觉。

为了支持正确的语法,你必须写一个全局函数:

classPoint{intx,y;public:Point(intx,inty):x(x),y(y){}// 声明全局函数为友元friendstd::ostream&operator<<(std::ostream&os,constPoint&p);};// 全局函数实现std::ostream&operator<<(std::ostream&os,constPoint&p){os<<"("<<p.x<<", "<<p.y<<")";// 直接访问私有变量 x, yreturnos;}

二、 友元破坏了封装吗?

这是一个常见的误区。事实上,友元不仅没有破坏封装,反而增强了封装。

  1. 主动权在类自己手中:你是被动地被“窥探”吗?不,是类自己声明了谁是它的朋友。权限的授予是单向且显式的。
  2. 避免暴露 Setter/Getter:如果没有友元,为了让外界(比如一个 Matrix 类)操作内部数据,你可能不得不被迫把数据设为public或者提供公有的set/get。这样所有人都能改它了。
  3. 精确授权:通过友元,你只把权限给了特定的“某个人”,而对全世界其他所有人依然保持封闭。

三、 友元的四条铁律(编译器视角)

友元的规则非常特殊,它们不遵循普通的继承逻辑:

  1. 友元关系不可逆:类 A 是类 B 的友元,并不代表 B 是 A 的友元。
  2. 友元关系不可传递:你的朋友的朋友,不是你的朋友。
  3. 友元关系不可继承:基类的友元在派生类中没有特权。老子的朋友不一定是儿子的朋友。
  4. 位置无关:在类定义中,friend声明写在publicprotected还是private下面效果是一样的,它不受访问控制符的影响。

四、 友元类与 Pimpl 模式

在复杂的工程中,我们常使用**友元类(Friend Class)**来实现“代理”或“逻辑分离”。
例如,一个Container类可能将Iterator类设为友元,这样迭代器就能高效地遍历容器的私有链表结构,而不需要将链表节点暴露给用户。


总结:有选择的信任

  • 友元是类的一种显式授权,用于解决某些全局函数或协作类必须访问私有数据的问题。
  • 它通过限制“知情权”的范围,避免了由于滥用public导致的封装劣化。
  • 准则:不到万不得已不用友元,但一旦需要,它比增加公有访问接口更安全。

下一篇预告:结束了继承与权限的探讨,我们要聊聊一个极其微小但影响深远的语法:为什么在虚函数后面写上= 0叫纯虚函数?而如果不小心写错了签名,编译器会怎么整你?

➡️《你真的了解C++吗》No.029:虚函数表中的“意外”——函数签名不匹配导致的重写失败。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/19 12:44:41

SGMICRO圣邦微 SGM58031XMS10G/TR MSOP10 模数转换芯片ADC

特性 单电源电压范围:3V至5.5V.PC总线电压范围:3V至5.5V 低静态电流: 连续模式:255pA(典型值) 掉电模式:0.8pA(典型值) 可选数据速率:6.25SPS至960SPS 输入多路复用器 4个单端输入或2个差分输入 内部可编程增益放大器(PGA) 内部电压参考与振荡器 可选数字比较器 2C兼容串行接口…

作者头像 李华
网站建设 2026/4/17 23:02:54

多路温度采集(有完整资料)

资料查找方式&#xff1a; 特纳斯电子&#xff08;电子校园网&#xff09;&#xff1a;搜索下面编号即可 编号&#xff1a; CP-51-2021-018 设计简介&#xff1a; 本设计是基于单片机的多路温度采集系统&#xff0c;主要实现以下功能&#xff1a; 可通过LCD1602显示温度和状…

作者头像 李华
网站建设 2026/4/17 19:25:32

AI短剧制作小程序源码系统,全智能创作系统源码全览

温馨提示&#xff1a;文末有资源获取方式当AI技术能将短剧制作成本压缩至传统方式的10%以下&#xff0c;当几个人几天内就能无拍摄完成一部剧集&#xff0c;“AI短剧”已从概念走进现实&#xff0c;成为普通人内容创业的黄金赛道。面对广阔前景与激烈竞争&#xff0c;一套功能强…

作者头像 李华
网站建设 2026/4/15 12:17:42

JS防爬虫3板斧

一、核心概念 1.1 JS防爬虫的基本原理 JS防爬虫是指通过JavaScript代码实现的反爬虫措施&#xff0c;其核心思想是&#xff1a; 浏览器执行验证&#xff1a;向客户端返回一段JavaScript代码&#xff0c;只有真实浏览器才能执行动态参数生成&#xff1a;通过JS计算生成动态token…

作者头像 李华
网站建设 2026/4/17 23:04:11

低功耗不是口号:LPWAN智能井盖如何实现3年免维护?

在智慧城市基础设施监测场景中&#xff0c;井盖看似微不足道&#xff0c;却是典型的“部署易、运维难”节点。它分布广、环境恶劣、无外部供电&#xff0c;且要求长期稳定运行——这对物联网终端的低功耗设计提出了极高要求。近年来&#xff0c;基于LPWAN&#xff08;低功耗广域…

作者头像 李华