当前位置: 首页 > news >正文

【Effective Modern C++】第三章 转向现代C++:10. 优先选用限域枚举,而非不限域枚举

使用限域enum来减少命名空间污染

通用规则:如果在一对大括号里声明一个名字,则该名字的可见性限定在括号括起来的作用域内。

但这个规则不适用于 C++98 风格的枚举类型中定义的枚举变量:枚举变量的属于包含着这个枚举类型的作用域,此作用域内不能有其他实体取相同的名字。

enum Color { black, white, red }; //black, white, red在Color所在的作用域 auto white = false; //错误! white早已在这个作用域中声明

枚举量的名字泄露到枚举类型所在作用域的现象被称为:不限范围的枚举类型(未限域enum)。

在 C++11 中,限定作用域的枚举类型(枚举类 / 限域enum)不会以这样的方式泄露名字:

enum class Color { black, white, red }; //black, white, red限制在Color域内 auto white = false; //没问题,域内没有其他“white” Color c = white; //错误,域中没有枚举名叫white Color c = Color::white; //没问题 auto c = Color::white; //也没问题(也符合5的建议)

限域枚举的枚举量是强类型的

强类型:语言会对类型兼容性做严格检查,禁止隐式的、不安全的类型转换,每个值都有明确且固定的类型,类型错误会被尽可能早地捕获(编译器 / 运行期)。

不限域枚举类型的枚举量可以隐式转换为整数类型(并能够从此处进一步转换到浮点类型),导致语义扭曲的代码合法:

enum Color { black, white, red }; //未限域enum std::vector<std::size_t>primeFactors(std::size_t x); //func返回x的质因子 Color c = red; if (c < 14.5) { // Color与double比较 (!) auto factors = primeFactors(c); // 向std::size_t参数传递Color (!) … }

从限定作用域的枚举类型到任何其他类型都不存在隐式转换路径,类型安全更优:

enum class Color { black, white, red }; //Color现在是限域enum Color c = Color::red; if (c < 14.5) { //错误!不能比较Color和double auto factors = primeFactors(c); //错误!不能向std::size_t参数传Color … } // 仅显式强制转换才允许(可控的类型转换) if (static_cast<double>(c) < 14.5) { auto factors = primeFactors(static_cast<std::size_t>(c)); }

限定作用域的枚举类型可以进行前置声明

前置声明的核心前提是编译器能确定枚举的底层类型(内存大小)

enum Color; // 错误!未限域enum无默认底层类型,编译器无法推断内存大小 enum class Color; // 没问题!限域enum默认底层类型为int,内存大小确定

底层类型与前置声明的完整规则

  1. 所有enum的底层类型是编译器为优化内存选择的整型(需覆盖所有枚举值的最小类型);
  2. 限域enum:默认底层类型为int(也可显式指定),因此总能直接前置声明
enum class Status; // 默认int,合法 enum class Status: std::uint32_t; // 显式指定底层类型,合法
  1. 未限域enum:无默认底层类型,仅显式指定底层类型时才能前置声明
enum Color: std::uint8_t; // 显式指定底层类型,合法
  1. 两者均可在定义时显式指定底层类型:
enum class Status: std::uint32_t { good = 0, failed = 1 }; enum Color: std::uint8_t { black, white, red };

未限域枚举的唯一实用场景:简化std::tuple字段访问

限域enum虽类型安全,但访问std::tuple字段时因无隐式转换,代码冗长;未限域enum可利用 “隐式转换为std::size_t” 简化模板实参传递:

// 定义tuple类型(存储用户名、邮箱、声望值) using UserInfo = std::tuple<std::string, std::string, std::size_t>; // 未限域enum:简洁访问tuple字段(依赖隐式转换为std::size_t) enum UserInfoFields { uiName, uiEmail, uiReputation }; UserInfo uInfo; auto val = std::get<uiEmail>(uInfo); // 直接用枚举成员作为模板实参 // 限域enum:无隐式转换,代码冗长 enum class UserInfoFields { uiName, uiEmail, uiReputation }; auto val = std::get<static_cast<std::size_t>(UserInfoFields::uiEmail)>(uInfo);

限域枚举访问 tuple 的优化方案:toUType constexpr 函数模板

为兼顾限域enum的类型安全与tuple访问的简洁性,可实现编译期生效toUType函数模板(转换枚举值为其底层类型):

C++11 版本:

template<typename E> constexpr typename std::underlying_type<E>::type toUType(E enumerator) noexcept { return static_cast<typename std::underlying_type<E>::type>(enumerator); }

C++14 简化版本(更简洁):

template<typename E> constexpr auto toUType(E enumerator) noexcept { return static_cast<std::underlying_type_t<E>>(enumerator); }

调用方式(兼顾安全与简洁):

auto val = std::get<toUType(UserInfoFields::uiEmail)>(uInfo);

总结

  • C++98风格的枚举类型称为不限范围的枚举类型。
  • 限定作用域的枚举类型仅在枚举类型内可见。它们只能通过cast强制类型转换以转换至其他类型。
  • 限定作用域的枚举类型和不限范围的枚举类型都支持底层类型指定。限域的枚举类型的默认底层类型是int,而不限域的枚举类型没有默认底层类型。
  • 限定作用域的枚举类型总是可以进行前置声明,而不限域的枚举类型却只有在指定了默认底层类型的前提下才可以进行前置声明。

原著在线阅读地址

http://www.jsqmd.com/news/298204/

相关文章:

  • 企业IT管理:如何合理设置文件夹权限?
  • Python应用日志管理:自动化按日期分割日志的完整指南
  • Docker Desktop虚拟化不支持导致启动失败解决方案
  • 《简易制作 Linux Shell:详细分析原理、设计与实践》
  • 2026年无机磨石厂商如何选?这份口碑榜单值得参考
  • 2026年新疆地区石英砂供应厂家实力推荐榜
  • 8 万个智能体 Skills 怎么选?skills.sh 排行榜帮你避开所有坑
  • 用AI快速验证你的AO3同人小说创意
  • NumPy零基础入门:AI助手教你玩转科学计算
  • 小白必看:HOSTS文件修改图文详解(含视频)
  • AI如何优化PING命令:智能网络诊断新方案
  • AI一键搞定TOMCAT安装:告别繁琐配置
  • 【开题答辩全过程】以 金太阳宠物用品网站为例,包含答辩的问题和答案
  • 【开题答辩全过程】以 基于web的高校水电费管理系统为例,包含答辩的问题和答案
  • 为初学者详细解析AXURE10的授权原理,提供官方和替代获取渠道,避免常见激活陷阱。
  • AI如何帮你快速掌握XP.1024新版本特性
  • 学霸同款2026 9款一键生成论文工具测评:本科生毕业论文必备清单
  • 2026年无机纤维喷涂工程优质厂商综合实力深度解析
  • 2026年成都石墨烯供热厂家甄选指南与优质企业推荐
  • 2026年徐州商务车租赁市场深度评测:五家主流服务商横向对比
  • 宁德会务接待平台选型指南与品牌实力解析
  • 聊聊合肥东辰艺考美术培训,师资、课程啥样,费用多少钱?
  • Docker新手必看:hello-world镜像报错完全解决指南
  • 讲讲廊坊大城短视频培训服务选择哪家好的秘诀
  • 2026视频制作新标杆:这家大型公司为何获业内盛赞?排名前十视频制作10年质保有保障
  • SpringBoot DFA 实现敏感词过滤
  • 美国大模型格局深度对比分析:Google、OpenAI、Anthropic、xAI与Adept的演进、竞争与未来
  • 美国主要大模型公司深度对比分析:Google、OpenAI、Anthropic、xAI与Adept全面研究
  • Google、OpenAI、Anthropic、xAI、Adept 五大美国主流大模型深度对比
  • SQLILABS通关效率提升300%:自动化测试技巧大全