c++怎么定义友元类与友元函数_c++ 访问私有成员权限设置【详解】

发布时间 - 2025-12-29 00:00:00    点击率:
友元类声明必须在类内部,且被授权类需提前声明;友元关系单向,不继承。如class B;前置声明后,A中friend class B;允许B访问A的私有成员,但A不能访问B的私有成员。

友元类声明必须写在类内部,且需提前声明被授权类

友元类不是“互相友好”,而是单向授权:A 类声明 friend class B; 后,B 类的成员函数可以访问 A 的私有/保护成员,但 A 不能因此访问 B 的私有成员。常见错误是把 friend class B; 写在 A 类定义之外,或在 B 尚未声明时就引用它。

正确做法:

  • B 类名必须在 A 类定义前已声明(可仅前置声明,无需完整定义)
  • friend class B; 必须出现在 A 类体内部,通常放在私有区或公有区均可,位置不影响权限
  • 友元关系不继承:若 C 继承 B,C 也不能自动访问 A 的私有成员
class B; // 前置声明

class A {
private:
    int secret = 42;
    friend class B; // ✅ 正确:B 被授权访问 A 的私有成员
};

class B {
public:
    void accessA(const A& a) {
        std::cout << a.secret << "\n"; // ✅ 可直接访问
    }
};

友元函数需在类内声明、在类外定义,声明时加 friend 关键字

友元函数不是类的成员,但它能访问该类所有私有/保护成员。关键点在于:类内只做声明(带 friend),不加作用域限定;定义必须在类外,且不能加 friend —— 加了会编译报错 "friend" declaration not in class scope

容易出错的情形:

  • 在类外定义时误写 friend void func(...); → 编译失败
  • 忘记在类内声明,只在类外定义 → 函数无权访问私有成员
  • 函数模板作友元时,未用 template 显式关联,导致实例化失败
class A {
private:
    double value = 3.14;
    friend void printSecret(const A& a); // ✅ 类内声明:带 friend,无实现
};

void printSecret(const A& a) { // ✅ 类外定义:不加 friend
    std::cout << a.value << "\n"; // ✅ 可访问
}

全局函数、类成员函数、重载操作符都可成为友元

友元不限于普通函数。你甚至可以把另一个类的某个成员函数设为友元(而非整个类),从而更精细地控制访问粒度。注意:被设为友元的成员函数所属类,必须在友元声明前完成前置声明或完整定义。

典型使用场景:

  • 重载 输出操作符:让 operator 访问类私有字段输出调试信息
  • 工厂类中某个创建函数需要读取目标类的私有构造参数
  • 两个紧密协作的类,仅允许对方特定方法访问自身敏感状态
class A {
private:
    std::string data = "internal";
    friend std::ostream& operator<<(std::ostream& os, const A& a); // ✅ 友元操作符
};

std::ostream& operator<<(std::ostream& os, const A& a) {
    return os << "[A:" << a.data << "]"; // ✅ 直接访问私有 data
}

友元破坏封装性,但某些场景下不可替代

友元不是设计缺陷的遮羞布,而是为解决真实耦合需求提供的机制。比如标准库中 std::stringstd::hash 的关系,或容器与其迭代器的协作,都依赖友元实现高效、安全的底层访问。

实际开发中要注意:

  • 优先考虑组合、接口抽象或提供受控的公有访问函数(如 get_secret_for_test()
  • 一旦引入友元,该类与友元之间的编译依赖即增强:修改 A 的私有布局,可能迫使所有友元重新编译
  • 单元测试中常借助友元函数访问私有逻辑,但应避免在生产代码中为测试而大量添加友元

最易被忽略的一点:友元声明不参与访问控制检查——即使写在 private: 区块里,它本身不是私有的,其他类仍能看到这个声明(只是无法调用)。这意味着友元关系是公开契约,不是隐藏实现细节。


# access  # c++  # String  # 封装  # 成员函数  # void  # 继承  # 接口  # 函数模板  # class  # private  # operator  # 作用域  # 写在  # 设为  # 不加  # 放在  # 出现在  # 均可  # 只在  # 时就  # 报错  # 而非 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel如何使用Eloquent ORM进行数据库操作?(CRUD示例)  Laravel如何集成Inertia.js与Vue/React?(安装配置)  如何获取免费开源的自助建站系统源码?  Laravel如何配置和使用缓存?(Redis代码示例)  网站建设保证美观性,需要考虑的几点问题!  Laravel与Inertia.js怎么结合_使用Laravel和Inertia构建现代单页应用  浏览器如何快速切换搜索引擎_在地址栏使用不同搜索引擎【搜索】  如何在腾讯云服务器上快速搭建个人网站?  东莞市网站制作公司有哪些,东莞找工作用什么网站好?  中山网站推广排名,中山信息港登录入口?  VIVO手机上del键无效OnKeyListener不响应的原因及解决方法  如何在万网自助建站平台快速创建网站?  手机软键盘弹出时影响布局的解决方法  Python3.6正式版新特性预览  制作电商网页,电商供应链怎么做?  济南网站建设制作公司,室内设计网站一般都有哪些功能?  详解Huffman编码算法之Java实现  Laravel如何配置.env文件管理环境变量_Laravel环境变量使用与安全管理  Laravel如何配置任务调度?(Cron Job示例)  Laravel如何生成URL和重定向?(路由助手函数)  国美网站制作流程,国美电器蒸汽鍋怎么用官方网站?  如何正确下载安装西数主机建站助手?  Laravel如何使用Vite进行前端资源打包?(配置示例)  详解ASP.NET 生成二维码实例(采用ThoughtWorks.QRCode和QrCode.Net两种方式)  Android GridView 滑动条设置一直显示状态(推荐)  详解一款开源免费的.NET文档操作组件DocX(.NET组件介绍之一)  Win11怎么设置默认图片查看器_Windows11照片应用关联设置  如何在阿里云虚拟主机上快速搭建个人网站?  如何快速生成橙子建站落地页链接?  湖南网站制作公司,湖南上善若水科技有限公司做什么的?  Laravel广播系统如何实现实时通信_Laravel Reverb与WebSockets实战教程  Laravel如何使用withoutEvents方法临时禁用模型事件  瓜子二手车官方网站在线入口 瓜子二手车网页版官网通道入口  怎么用AI帮你为初创公司进行市场定位分析?  Laravel中Service Container是做什么的_Laravel服务容器与依赖注入核心概念解析  最好的网站制作公司,网购哪个网站口碑最好,推荐几个?谢谢?  Laravel Session怎么存储_Laravel Session驱动配置详解  Laravel如何使用Passport实现OAuth2?(完整配置步骤)  b2c电商网站制作流程,b2c水平综合的电商平台?  网站页面设计需要考虑到这些问题  ,网页ppt怎么弄成自己的ppt?  Laravel用户密码怎么加密_Laravel Hash门面使用教程  Laravel如何实现URL美化Slug功能_Laravel使用eloquent-sluggable生成别名【方法】  Laravel如何处理JSON字段的查询和更新_Laravel JSON列操作与查询技巧  实例解析angularjs的filter过滤器  rsync同步时出现rsync: failed to set times on “xxxx”: Operation not permitted  Laravel Admin后台管理框架推荐_Laravel快速开发后台工具  如何在云主机快速搭建网站站点?  零服务器AI建站解决方案:快速部署与云端平台低成本实践  Laravel如何正确地在控制器和模型之间分配逻辑_Laravel代码职责分离与架构建议