首页 游戏 软件 资讯 排行榜 专题
首页
编程语言
c#如何定义枚举_c#定义枚举完整教程与代码实例

c#如何定义枚举_c#定义枚举完整教程与代码实例

热心网友
93
转载
2026-05-05

C#枚举:从定义到实战,避开那些“坑”

c#如何定义枚举_c#定义枚举完整教程与代码实例

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

开门见山,先说核心结论:在C#中定义一个枚举,基础语法确实简单,enum 名称 { 值1, 值2 } 即可完成。但如果你只停留在这一步,很可能在后续的反序列化、跨平台通信等复杂场景中“踩坑”。真正需要开发者深入关注的,是底层存储类型、显式赋值策略、可空性处理以及序列化行为这些关键细节。

枚举的底层类型和显式赋值:不只是默认的int

许多开发者可能没有意识到,C#中enum默认的底层存储类型是int。这意味着每个枚举值默认占用4个字节。但在资源敏感或系统交互的场景下,这可能造成内存浪费。你可以显式指定为byteshortlong等其他整数类型,这直接关系到内存占用、性能以及系统间的互操作性——例如与C/C++结构体进行内存对齐,或者映射到数据库的tinyintsmallint字段。

那么,问题通常会如何暴露呢?一种典型情况是,当你尝试将一个byte类型的枚举值不加转换地赋值给int变量时,可能会遇到InvalidCastException。另一种常见陷阱是在使用System.Text.Json.JsonSerializer进行序列化时,如果枚举值超出了int的范围,而你又未调整底层类型,序列化过程就可能失败。

  • 场景一:追求极致空间效率或对接硬件协议。此时应优先考虑enum Status : byte { Off = 0, On = 1 }。仅用一个字节,既清晰又节省内存。
  • 场景二:需要容纳超大数值标识。例如希望用枚举表示Unix时间戳级别的ID,就必须使用enum EventId : long { Created = 1L }来提供足够的数值空间。
  • 关于赋值规则:如果不进行显式赋值,编译器会从0开始自动递增。但一旦你为其中一项指定了值(例如Running = 5),后续的项则会在此基础上继续按+1的规则递推(例如Stopped = 6)。理解并掌握这个规则至关重要。

字符串解析:如何优雅地避开ArgumentException?

将字符串转换为枚举值,Enum.ParseEnum.TryParse是最常用的方法。但它们的默认行为较为严格:区分大小写,并且不接受空格或连字符等特殊字符。在生产环境中,用户输入或来自第三方API的JSON数据字段名往往格式不规范,直接进行硬解析极易导致程序抛出异常。

举例说明:你的Web API接收一个查询参数?state=ready-to-run,但枚举定义中对应的项是ReadyToRun。中间的连字符就成了解析失败的“元凶”。

  • 首选方案:使用TryParse并忽略大小写。务必写成Enum.TryParse(str, ignoreCase: true, out var result)。如果不传递ignoreCase: true参数,像"readytorun"这样全小写的字符串就会解析失败。
  • 处理特殊字符:如果字符串中包含连字符、下划线、空格等,内置方法无法直接识别。稳妥的做法是预先进行字符串清洗:str.Replace("-", "").Replace("_", "").Replace(" ", ""),然后再尝试解析。
  • 重要安全原则:永远不要使用Enum.Parse方法去处理不可信的外部输入源,因为它会直接抛出ArgumentExceptionArgumentNullException。相比之下,TryParse返回一个布尔值来指示成功与否,并提供out参数输出结果,控制流程更安全、更优雅。

Flags枚举:位运算背后的“潜规则”

为枚举加上[Flags]特性,并不意味着你可以随意组合任意值。这里存在一个关键前提:每一个单独定义的枚举值,其数值都必须是2的幂(即1, 2, 4, 8, 16…)。如果违反了这个二进制位独立的原则,那么.ToString()HasFlag()的检查逻辑以及JSON序列化的输出结果都可能出现难以预料的错误,给调试带来巨大困扰。

典型的现象是:当你组合FlagAFlagBMyFlags f = FlagA | FlagB;),输出看起来正常("FlagA, FlagB")。但如果FlagB的值被误设为3(它不是2的幂),那么f.ToString()可能返回一个空字符串,或者一个错误的、无法识别的组合名称。

  • 定义时的最佳实践:强制使用左移位运算符来定义值,例如Read = 1 << 0, Write = 1 << 1, Execute = 1 << 2。这种方式能一目了然地确保每个枚举值都占据一个独立的二进制位,符合Flags枚举的设计初衷。
  • 检查标志位:推荐使用高效的位与运算(f & Read) == Read来判断是否包含某个标志。传统的f.HasFlag(Read)方法由于涉及装箱和类型检查,性能开销较大,并且在 .NET 6+ 中已被标记为过时(obsolete)。
  • 序列化输出控制:默认情况下,Flags枚举在JSON序列化中会输出其对应的整数值。如果你希望输出更易读的、逗号分隔的名称列表(例如"Read, Write"),则需要配置序列化器使用JsonStringEnumConverter,并注意设置AllowIntegerValues = false以防止数值形式的意外输入。

总而言之,C#枚举看似语法简单,但在跨框架(如 .NET Framework 与 .NET 5/6/7/8 之间)、跨序列化器(System.Text.Json 与 Newtonsoft.Json)、跨语言(如gRPC的proto枚举映射)的复杂场景下,有三个核心问题最容易被忽略:底层存储类型不一致、命名策略(大小写、分隔符)不匹配,以及Flags枚举的值非法(非2的幂)。尤其是在重构旧项目或对接遗留系统时,对这些技术细节多留一份心,能有效规避潜在风险,显著提升代码的健壮性和可维护性。

来源:https://www.php.cn/faq/2333495.html
免责声明: 游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

相关攻略

c#如何绘制图形_c#绘制图形的正确用法与注意事项
编程语言
c#如何绘制图形_c#绘制图形的正确用法与注意事项

C 绘图避坑指南:从Graphics来源到DPI适配的实战要点 在C 中进行图形绘制,一个看似简单的DrawRectangle背后,往往藏着好几个“坑”。Graphics对象不能直接new,否则要么直接报错,要么静默失败——所有绘图操作都必须基于合法的来源。这可以说是入门绘图的第一条铁律。 Grap

热心网友
05.05
VSCode怎么搭建Unity 3D的C#脚本编写环境并解决找不到引用的问题
编程语言
VSCode怎么搭建Unity 3D的C#脚本编写环境并解决找不到引用的问题

VSCode怎么搭建Unity 3D的C 脚本编写环境并解决找不到引用的问题 在Unity开发中,用VSCode写C 脚本时遇到“找不到引用”的红色波浪线,这事儿确实挺让人头疼的。别急,这通常不是代码逻辑问题,而是开发环境之间的“沟通”出了岔子。下面咱们就来逐一拆解最常见的几个原因和对应的解决方案。

热心网友
05.04
C#如何使用Record类型_C#不可变数据模型特性解析【极简】
编程语言
C#如何使用Record类型_C#不可变数据模型特性解析【极简】

C Record类型:不可变数据容器的正确打开方式 先明确一个核心认知:C 中的Record类型,本质上是一个“省心”的不可变数据容器。它不是什么更高级的class,而是编译器帮你自动生成值相等性、ToString、GetHashCode以及with表达式的语法糖。用对了,它能帮你省掉80%的数据

热心网友
05.03
C#如何获取硬件信息_C# WMI读取CPU与硬盘序列号【进阶】
编程语言
C#如何获取硬件信息_C# WMI读取CPU与硬盘序列号【进阶】

WMI无法稳定读取现代CPU与NVMe硬盘序列号?问题不在代码,而在硬件与系统本身 一个常见的开发误区是:用WMI读取CPU和硬盘序列号,结果发现拿不到、拿不准或者拿到一堆乱码。问题往往不在于你的代码写错了,而是系统或固件层面,压根就没把这个“身份证号”暴露给你。 为什么 Win32_Process

热心网友
05.02
C#怎么防止UI线程假死_C#耗时操作放入后台线程更新UI【核心】
编程语言
C#怎么防止UI线程假死_C#耗时操作放入后台线程更新UI【核心】

C 怎么防止UI线程假死_C 耗时操作放入后台线程更新UI【核心】 耗时操作必须离开 UI 线程,否则假死不可避免 —— 这不是优化建议,而是 WinForms WPF 的运行铁律。 为什么直接在 Button_Click 里调用 Thread Sleep 就卡死? 道理其实很简单:UI 线程身兼数

热心网友
05.01

最新APP

宝宝过生日
宝宝过生日
应用辅助 04-07
台球世界
台球世界
体育竞技 04-07
解绳子
解绳子
休闲益智 04-07
骑兵冲突
骑兵冲突
棋牌策略 04-07
三国真龙传
三国真龙传
角色扮演 04-07

热门推荐

怎样用telnet管理网络设备
编程语言
怎样用telnet管理网络设备

使用Telnet管理网络设备:一份实用指南 在网络设备管理的众多工具中,Telnet堪称一位“资深元老”。它以简洁、直接的方式,让管理员能够从远程便捷地登录路由器或交换机的命令行界面。然而,必须首先明确一个关键点:Telnet协议本身缺乏安全保障,其传输的所有数据,包括用户名和密码,均以明文形式进行

热心网友
05.05
如何用telnet调试网络应用
编程语言
如何用telnet调试网络应用

使用Telnet调试网络应用:快速定位连接与协议问题 在网络应用开发与日常运维中,高效排查故障是必备技能。Telnet作为经典的网络协议工具,凭借其简洁的命令行交互方式,至今仍是测试端口连通性、验证服务响应及手动调试文本协议的实用选择。它无需图形界面,直接通过命令行揭示网络层的真实状态,是工程师手中

热心网友
05.05
如何利用cpustat进行系统监控
编程语言
如何利用cpustat进行系统监控

全面掌握系统性能:使用 cpustat 工具进行专业级 CPU 监控 在 Linux 系统性能优化与故障诊断过程中,CPU 使用率是至关重要的核心指标。作为 sysstat 工具集的重要组成部分,cpustat 命令为系统管理员和开发者提供了一种直接、高效且深入的 CPU 监控解决方案。本文将详细介

热心网友
05.05
cpustat如何辅助进行性能调优
编程语言
cpustat如何辅助进行性能调优

掌握cpustat:Linux系统性能监控与CPU调优的必备工具 在Linux服务器性能优化与故障排查过程中,CPU资源的使用状况通常是首要分析目标。除了广为人知的top和htop命令,cpustat是一款同样强大却常被忽略的专业级CPU监控利器。作为sysstat工具集的核心组件之一,它能够实时采

热心网友
05.05
如何用cpustat查看进程CPU使用情况
编程语言
如何用cpustat查看进程CPU使用情况

使用 cpustat 监控进程 CPU 使用情况 在 Linux 系统性能调优与故障排查过程中,精准监控 CPU 使用率是至关重要的基础技能。cpustat 作为 sysstat 工具集的核心组件之一,专门为深入洞察 CPU 资源分配与消耗而设计。它提供了超越常规系统监控命令的、聚焦于处理器性能的详

热心网友
05.05