从C++17、C++20到C++23,不断进阶的枚举类!

C语言与CPP编程 2024-09-10 09:01

本文经授权转自公众号CSDN(ID:CSDNnews)

作者 | Bartłomiej Filipek

随着 C++ 标准的不断演进,枚举类型(enum class)作为一种重要的数据结构,在 C++ 社区中扮演着越来越重要的角色。从 C++17 到 C++23,我们见证了枚举类型的多项改进和完善,这些变化不仅增强了语言本身的表达能力,也为开发者提供了更强大、更安全的编程工具。

原文链接:https://www.cppstories.com/2024/enum-improvements/


C++语言的演进不断带来强大的新特性,提升了代码的安全性、可读性和可维护性。在这些改进中,我们见证了从 C++17、C++20 到 C++23 中对 enum class 功能的修改和扩展。在这篇文章中,我们将探讨这些进展,重点介绍 C++17 中的初始化改进、C++20 中引入的 using enum 关键字,以及 C++23 中的 std::to_underlying 实用工具

1、enum class 简介

在深入了解这些改进之前,让我们先简要回顾一下 enum class 是什么。enum class(限定作用域的枚举)提供了一种类型安全的方式来定义一组命名常量。与传统的(无作用域)枚举不同,enum class 不会隐式转换为整数或其他类型,从而防止了意外的误用。下面是一个基本示例:

#include 
enum class Color { Red, Green, Blue};
int main() { Color color = Color::Red;
if (color == Color::Red) std::cout << "The color is red.\n";
color = Color::Blue;
if (color == Color::Blue) std::cout << "The color is blue.\n";
// std::cout << color; // error, no matching << operator // int i = color; // error: cannot convert}

注意,在 main 函数的末尾有两行代码。由于没有隐式转换为整数类型,因此会出现编译器错误。

作为对比,下面是一个类似的例子,但使用的是无作用域枚举:

#include 
enum Color { Red, Green, Blue};
int main() { Color color = Red;
if (color == Red) std::cout << "The color is red.\n";
color = Blue;
if (color == Blue) std::cout << "The color is blue.\n";
std::cout << color; // fine, prints integer value! int i = color; // fine, can convert...}

简而言之,enum class 为所有枚举值提供了单独的作用域,同时也加强了类型安全。没有隐式的整数转换,这样你就能更好地控制设计。

以上的基础部分很简单,接下来让我们来看看最新 C++ 版本中的一些实用改进。


2、C++17:使用大括号初始化基础类型

有时候,enum class 可能显得过于限制,某些情况下的转换可能会很方便。

在 C++17 中,P0138 提案被接受,以下是其中的一个示例:

enum class Handle : uint32_t { Invalid = 0 }; Handle h { 42 }; // OK

简而言之,当你使用 enum class 来定义强类型时,允许从基础类型进行初始化而不产生任何错误——这在 C++17 之前是无法实现的。

这个变化仍能确保枚举仍然是安全的,因为它们只能用于统一/大括号初始化。请看看下面的代码:

#include 
enum class Handle : uint32_t { Invalid = 0 };
void process(Handle h) {
}
int main() { Handle h { 42 }; // OK
// process({10}); // error process(Handle{10});}

你不能直接将 {10} 作为 process 函数的参数,仍需要明确指定类型。

在 C++14 中,你可以使用 process(static_cast(10));——如你所见,C++17 版本的改进要好得多。


3、C++20:使用 using enum

C++20 引入了 using enum 语法,这个特性允许你将一个枚举的所有枚举值引入当前作用域,同时不失去作用域枚举的优点。请看下面的示例:

enum class ComputeStatus {    Ok,    Error,    FileError,    NotEnoughMemory,    TimeExceeded,    Unknown};

在早期的 C++ 版本中,使用这些枚举值时,需要使用枚举类的名称进行限定:

ComputeStatus s = ComputeStatus::NotEnoughMemory;

C++20 通过 using enum 声明简化了这一点:

int main() {    using enum ComputeStatus;    ComputeStatus s = NotEnoughMemory;}

上面的简单代码可能没什么实际意义,但看看下面这个例子:

int main() {        ComputeStatus s = ComputeStatus::Ok;    switch (s) {        case ComputeStatus::Ok:             std::cout << "ok"; break;        case ComputeStatus::Error:             std::cout << "Error"; break;        case ComputeStatus::FileError:             std::cout << "FileError"; break;        case ComputeStatus::NotEnoughMemory:             std::cout << "NotEnoughMemory"; break;        case ComputeStatus::TimeExceeded:             std::cout << "Time..."; break;        default: std::cout << "unknown...";    }}

我们可以将其转换为如下形式:

int main() {        ComputeStatus s = ComputeStatus::Ok;    switch (s) {        using enum ComputeStatus;  // << <<        case Ok:             std::cout << "ok"; break;        case Error:             std::cout << "Error"; break;        case FileError:             std::cout << "FileError"; break;        case NotEnoughMemory:             std::cout << "NotEnoughMemory"; break;        case TimeExceeded:             std::cout << "Time..."; break;        default: std::cout << "unknown...";    }}

或者,也可以看看下面这个例子:

struct ComputeEngine {    enum class ComputeStatus {        Ok,        Error,        FileError,        NotEnoughMemory,        TimeExceeded,        Unknown    };    using enum ComputeStatus;};
int main() { ComputeEngine::ComputeStatus s = ComputeEngine::Ok;}

你可以将所有枚举值引入 ComputeEngine 的作用域中,同时享受 enum class 带来的类型安全特性。

C++20 的这一改进使代码更加简洁,并减少了冗余,尤其是在某个作用域内频繁使用多个枚举值的情况下。它提供了一种更加流畅和可读的方式,同时不牺牲作用域枚举所提供的类型安全性。


4、C++23:std::to_underlying

C++23 通过引入 std::to_underlying 进一步增强了 enum class 的可用性,这个实用函数可以将枚举值转换为其基础的整型类型,这个特性解决了将枚举值转换为整数以用于存储、比较或与其他期望整型的 API 交互的常见需求。

这个想法最早出现在 Scott Meyers 的经典著作《Effective Modern C++: 42 Specific Ways to Improve Your Use of C++11 and C++14》一书中。终于在 C++23 中,我们可以享受到这个被标准化的功能。

在 C++23 之前,将枚举转换为其基础类型需要显式的类型转换:

enum class Permissions : uint8_t {    Execute = 1,  Write = 2,    Read = 4};
uint8_t value = static_cast<uint8_t>(Permissions::Read);

而有了 std::to_underlying,这个转换变得更加直接和清晰:

#include 
int main() { Permissions p = Permissions::Read; auto value = std::to_underlying(p); // C++23}

std::to_underlying 函数提高了代码的可读性,并减少了与类型转换相关的样板代码。它还明确了意图,使得人们能一目了然地知道这是在获取枚举的基础值。


5、未来的改进

接下来,一个即将到来的重要更新可能是支持 C++26 反射——可查看提案 P2996(虽然这个提案还未被接受,但预计很快就会获得批准)。反射带来了许多激动人心的可能性,比如将枚举转换为字符串的能力。

请看提案中的这个例子:

template <typename E>  requires std::is_enum_vconstexpr std::string enum_to_string(E value) {  template for (constexpr auto e : std::meta::enumerators_of(^E)) {    if (value == [:e:]) {      return std::string(std::meta::name_of(e));    }  }  return "";}
enum Color { red, green, blue };static_assert(enum_to_string(Color::red) == "red");static_assert(enum_to_string(Color(42)) == "");

当然,你也可以不必等到 C++26,依赖第三方库就能体验这个功能,如 Neargye/magic_enum @GitHub。   

本文转自公众号“CSDN”,ID:CSDNnews

分享个群友推荐的招聘类小程序

这里分享一个交流群群友推荐的招聘小程序,其中有很多二三线城市,比如赤峰、保定、阿克苏之类的三四线城市,还支持按照岗位、地点和薪资要求来找合适的岗位。

经常遇到一些想回老家或者二三线城市的同学苦于没有合适的去处,打开boss直聘和猎聘网这些招聘类软件,结果发现好的岗位基本都集中在一线城市,很少看到那种有二三线城市的招聘岗位。

比如说我是一个河南人,以后想回老家省会郑州求职的话,我可以筛选岗位为Linux C/C++开发、地点为郑州、薪资为12-18k的工作,那么直接筛选即可。

昨天已经在几个交流群中分享了一下,今天也给大家分享一二,扫描下方二维码注册个账号即可体验。

这几年行情不好,有好的工作机会还是一起分享,报团取暖才是王道。如果有公司负责人之类的朋友想要招聘,可以直接私聊我,必须帮忙扩散分享~

EOF

日常分享C/C++、计算机学习经验、工作体会,欢迎点击此处查看我以前的学习笔记&经验&分享的资源。

我组建了一些社群一起交流,群里有大牛也有小白,如果你有意可以一起进群交流。

欢迎你添加我的微信,我拉你进技术交流群。此外,我也会经常在微信上分享一些计算机学习经验以及工作体验,还有一些内推机会

加个微信,打开另一扇窗

经常遇到有读者后台私信想要一些编程学习资源,这里分享 1T 的编程电子书、C/C++开发手册、Github上182K+的架构路线图、LeetCode算法刷题笔记等精品学习资料,点击下方公众号会回复"编程"即可免费领取~

感谢你的分享,点赞,在看三  

C语言与CPP编程 C语言/C++开发,C语言/C++基础知识,C语言/C++学习路线,C语言/C++进阶,数据结构;算法;python;计算机基础等
评论
  • 根据Global Info Research(环洋市场咨询)项目团队最新调研,预计2030年全球无人机电池和电源产值达到2834百万美元,2024-2030年期间年复合增长率CAGR为10.1%。 无人机电池是为无人机提供动力并使其飞行的关键。无人机使用的电池类型因无人机的大小和型号而异。一些常见的无人机电池类型包括锂聚合物(LiPo)电池、锂离子电池和镍氢(NiMH)电池。锂聚合物电池是最常用的无人机电池类型,因为其能量密度高、设计轻巧。这些电池以输出功率大、飞行时间长而著称。不过,它们需要
    GIRtina 2025-01-13 10:49 164浏览
  • 随着通信技术的迅速发展,现代通信设备需要更高效、可靠且紧凑的解决方案来应对日益复杂的系统。中国自主研发和制造的国产接口芯片,正逐渐成为通信设备(从5G基站到工业通信模块)中的重要基石。这些芯片凭借卓越性能、成本效益及灵活性,满足了现代通信基础设施的多样化需求。 1. 接口芯片在通信设备中的关键作用接口芯片作为数据交互的桥梁,是通信设备中不可或缺的核心组件。它们在设备内的各种子系统之间实现无缝数据传输,支持高速数据交换、协议转换和信号调节等功能。无论是5G基站中的数据处理,还是物联网网关
    克里雅半导体科技 2025-01-10 16:20 433浏览
  • 新年伊始,又到了对去年做总结,对今年做展望的时刻 不知道你在2024年初立的Flag都实现了吗? 2025年对自己又有什么新的期待呢? 2024年注定是不平凡的一年, 一年里我测评了50余块开发板, 写出了很多科普文章, 从一个小小的工作室成长为科工公司。 展望2025年, 中国香河英茂科工, 会继续深耕于,具身机器人、飞行器、物联网等方面的研发, 我觉得,要向未来学习未来, 未来是什么? 是掌握在孩子们生活中的发现,和精历, 把最好的技术带给孩子,
    丙丁先生 2025-01-11 11:35 447浏览
  • 在不断发展的电子元件领域,继电器——作为切换电路的关键设备,正在经历前所未有的技术变革。固态继电器(SSR)和机械继电器之间的争论由来已久。然而,从未来发展的角度来看,固态继电器正逐渐占据上风。本文将从耐用性、速度和能效三个方面,全面剖析固态继电器为何更具优势,并探讨其在行业中的应用与发展趋势。1. 耐用性:经久耐用的设计机械继电器:机械继电器依靠物理触点完成电路切换。然而,随着时间的推移,这些触点因电弧、氧化和材料老化而逐渐磨损,导致其使用寿命有限。因此,它们更适合低频或对切换耐久性要求不高的
    腾恩科技-彭工 2025-01-10 16:15 97浏览
  • 随着数字化的不断推进,LED显示屏行业对4K、8K等超高清画质的需求日益提升。与此同时,Mini及Micro LED技术的日益成熟,推动了间距小于1.2 Pitch的Mini、Micro LED显示屏的快速发展。这类显示屏不仅画质卓越,而且尺寸适中,通常在110至1000英寸之间,非常适合应用于电影院、监控中心、大型会议、以及电影拍摄等多种室内场景。鉴于室内LED显示屏与用户距离较近,因此对于噪音控制、体积小型化、冗余备份能力及电气安全性的要求尤为严格。为满足这一市场需求,开关电源技术推出了专为
    晶台光耦 2025-01-13 10:42 487浏览
  • ARMv8-A是ARM公司为满足新需求而重新设计的一个架构,是近20年来ARM架构变动最大的一次。以下是对ARMv8-A的详细介绍: 1. 背景介绍    ARM公司最初并未涉足PC市场,其产品主要针对功耗敏感的移动设备。     随着技术的发展和市场需求的变化,ARM开始扩展到企业设备、服务器等领域,这要求其架构能够支持更大的内存和更复杂的计算任务。 2. 架构特点    ARMv8-A引入了Execution State(执行状
    丙丁先生 2025-01-12 10:30 451浏览
  •   在信号处理过程中,由于信号的时域截断会导致频谱扩展泄露现象。那么导致频谱泄露发生的根本原因是什么?又该采取什么样的改善方法。本文以ADC性能指标的测试场景为例,探讨了对ADC的输出结果进行非周期截断所带来的影响及问题总结。 两个点   为了更好的分析或处理信号,实际应用时需要从频域而非时域的角度观察原信号。但物理意义上只能直接获取信号的时域信息,为了得到信号的频域信息需要利用傅里叶变换这个工具计算出原信号的频谱函数。但对于计算机来说实现这种计算需要面对两个问题: 1.
    TIAN301 2025-01-14 14:15 97浏览
  • 流量传感器是实现对燃气、废气、生活用水、污水、冷却液、石油等各种流体流量精准计量的关键手段。但随着工业自动化、数字化、智能化与低碳化进程的不断加速,采用传统机械式检测方式的流量传感器已不能满足当代流体计量行业对于测量精度、测量范围、使用寿命与维护成本等方面的精细需求。流量传感器的应用场景(部分)超声波流量传感器,是一种利用超声波技术测量流体流量的新型传感器,其主要通过发射超声波信号并接收反射回来的信号,根据超声波在流体中传播的时间、幅度或相位变化等参数,间接计算流体的流量,具有非侵入式测量、高精
    华普微HOPERF 2025-01-13 14:18 462浏览
  • 电动汽车(EV)正在改变交通运输,为传统内燃机提供更清洁、更高效的替代方案。这种转变的核心是电力电子和能源管理方面的创新,而光耦合器在其中发挥着关键作用。这些不起眼的组件可实现可靠的通信、增强安全性并优化电动汽车系统的性能,使其成为正在进行的革命中不可或缺的一部分。光耦合器,也称为光隔离器,是一种使用光传输电信号的设备。通过隔离高压和低压电路,光耦合器可确保安全性、减少干扰并保持信号完整性。这些特性对于电动汽车至关重要,因为精确控制和安全性至关重要。 光耦合器在电动汽车中的作用1.电池
    腾恩科技-彭工 2025-01-10 16:14 71浏览
  • PNT、GNSS、GPS均是卫星定位和导航相关领域中的常见缩写词,他们经常会被用到,且在很多情况下会被等同使用或替换使用。我们会把定位导航功能测试叫做PNT性能测试,也会叫做GNSS性能测试。我们会把定位导航终端叫做GNSS模块,也会叫做GPS模块。但是实际上他们之间是有一些重要的区别。伴随着技术发展与越发深入,我们有必要对这三个词汇做以清晰的区分。一、什么是GPS?GPS是Global Positioning System(全球定位系统)的缩写,它是美国建立的全球卫星定位导航系统,是GNSS概
    德思特测试测量 2025-01-13 15:42 466浏览
  • 随着全球向绿色能源转型的加速,对高效、可靠和环保元件的需求从未如此强烈。在这种背景下,国产固态继电器(SSR)在实现太阳能逆变器、风力涡轮机和储能系统等关键技术方面发挥着关键作用。本文探讨了绿色能源系统背景下中国固态继电器行业的前景,并强调了2025年的前景。 1.对绿色能源解决方案日益增长的需求绿色能源系统依靠先进的电源管理技术来最大限度地提高效率并最大限度地减少损失。固态继电器以其耐用性、快速开关速度和抗机械磨损而闻名,正日益成为传统机电继电器的首选。可再生能源(尤其是太阳能和风能
    克里雅半导体科技 2025-01-10 16:18 319浏览
  • 01. 什么是过程能力分析?过程能力研究利用生产过程中初始一批产品的数据,预测制造过程是否能够稳定地生产符合规格的产品。可以把它想象成一种预测。通过历史数据的分析,推断未来是否可以依赖该工艺持续生产高质量产品。客户可能会要求将过程能力研究作为生产件批准程序 (PPAP) 的一部分。这是为了确保制造过程能够持续稳定地生产合格的产品。02. 基本概念在定义制造过程时,目标是确保生产的零件符合上下规格限 (USL 和 LSL)。过程能力衡量制造过程能多大程度上稳定地生产符合规格的产品。核心概念很简单:
    优思学院 2025-01-12 15:43 496浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦