您的位置:首页 >c#如何定义枚举_c#定义枚举完整教程与代码实例
发布于2026-05-02 阅读(0)
扫一扫,手机访问

开门见山,先说核心结论:在C#里定义一个枚举,语法上确实简单,enum 名称 { 值1, 值2 } 就能搞定。但如果你只停留在这一步,那很可能在后续的反序列化、跨平台通信等场景里“踩坑”。真正要关注的,是底层类型、显式赋值、可空性以及序列化行为这些细节。
很多人可能没意识到,enum 默认的底层类型是 int。这意味着每个枚举值都占4个字节。但在某些场景下,这会造成浪费。你可以显式指定为 byte、short、long 等类型,这直接关系到内存占用和系统间的互操作性——比如与C/C++结构体对齐,或者映射到数据库的 tinyint 字段。
那么,问题通常会怎么暴露出来呢?一种常见情况是,当你试图将一个 byte 类型的枚举值不加转换地赋给 int 变量时,可能会遇到 Invalid cast exception。另一种情况是使用 JsonSerializer 进行序列化时,如果枚举值超出了 int 的范围,而你又没调整底层类型,序列化就会失败。
enum Status : byte { Off = 0, On = 1 }。一个字节,清晰又省地儿。enum EventId : long { Created = 1L } 来撑起场面。Running = 5),后续的项则会在此基础上继续按+1递推(例如 Stopped = 6)。这个规则需要心里有数。把字符串转换成枚举,Enum.Parse 和 Enum.TryParse 是最常用的工具。但它们的默认行为有点“死板”:区分大小写,并且不接受空格或连字符。在生产环境中,用户输入或第三方JSON数据的字段名往往不那么规范,直接硬解析很容易导致程序崩溃。
举个例子:你的API接收一个查询参数 ?state=ready-to-run,但枚举定义里对应的项是 ReadyToRun。这中间的连字符就成了拦路虎。
Enum.TryParse(str, ignoreCase: true, out var result) 。如果不传 ignoreCase: true,像 "readytorun" 这样的字符串就会解析失败。str.Replace("-", "").Replace("_", ""),然后再尝试解析。Enum.Parse 去处理不可信的输入源,因为它会抛出 ArgumentException 或 ArgumentNullException。相比之下,TryParse 返回一个布尔值来表示成功与否,控制起来更安全、更优雅。给枚举加上 [Flags] 特性,并不意味着你可以随意组合值。这里有个关键前提:每一个定义的枚举值,都必须是2的幂(即1, 2, 4, 8…)。如果违反了这个规则,那么 .ToString()、HasFlag() 以及JSON序列化的结果都可能出现逻辑错误,让你摸不着头脑。
典型的现象是:当你组合 FlagA 和 FlagB(MyFlags f = FlagA | FlagB;),输出看起来正常("FlagA, FlagB")。但如果 FlagB 的值是3(不是2的幂),那么 f.ToString() 可能返回一个空字符串,或者错误的组合名称。
Read = 1 << 0, Write = 1 << 1, Execute = 1 << 2。这样能一目了然地确保每个值都是独立的二进制位。(f & Read) == Read 来判断是否包含某个标志。传统的 f.HasFlag(Read) 方法性能较差,并且在 .NET Core 6+ 中已被标记为过时(obsolete)。"Read, Write"),则需要配置 JsonStringEnumConverter,并记得设置 AllowIntegerValues = false。总而言之,枚举看似简单,但在跨框架(如 .NET Framework 与 .NET 5+ 之间)、跨序列化器(System.Text.Json 与 Newtonsoft.Json)、跨语言(如gRPC的proto枚举映射)的场景下,有三个问题最容易被忽略:底层类型不一致、命名策略不匹配,以及Flags枚举的值非法。尤其是在重构旧项目或对接遗留系统时,对这些细节多留一份心,能省去很多调试的麻烦。
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
正版软件
正版软件
正版软件
正版软件
正版软件
1
2
3
7
9