- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
在阅读某人的源代码时,我遇到了一个 C# 片段,其行为让我感到困惑。在我开始提问之前,这里有一个改编的例子:
枚举:
private enum Seasons
{
Spring,
Summer,
Autumn,
Winter
}
用法:
foreach(Seasons value in Enum.GetValues(typeof(Seasons)))
{
Console.WriteLine(String.Format("{0} {1}", value, (int) value));
}
输出看起来像这样:
Spring 0
Summer 1
Autumn 2
Winter 3
我的问题:
这在 C# 中是如何工作的?返回值来自:
Enum.GetValues(typeof(Seasons))
是一个 int 数组,其中填充了 0-3 的值。当它被转换为 Seasons 枚举时,我首先预计它会失败。
我认为由于 enum 是一个引用类型,在 String.Format() 参数“值”上,将调用 ToString() 方法,它将返回它的名字。但是当转换为 int 时,它将使用该“命名”值上的属性并返回 int 值。
如果有人能告诉我这是如何在幕后运作的,我会很高兴。我查看了文档,但还无法解决。
谢谢...
谢谢大家的回答!
但是......如果有人能多谈谈 C# 内部结构,我将不胜感激。在 Java 中,枚举将是一个真正的引用类型,在内存中,它将被视为指向另一个存储整数的内存空间的引用。像这样非常非常简单。
--------- -----------------
| | Reference |int Spring |
|Seasons|-------------> |int Summer |
| | |int Autumn |
--------- |int Winter |
-----------------
在我看来,在 C# 中,编译器将其转换为类似 public const 的东西(在我的例子中是 int)。请不要扔石头...
最佳答案
Seasons 类型的变量可以被赋予基础类型范围内的任何值(整数在这种情况下)。值甚至不限于枚举的命名常量。因此,通过将整数值 5
转换为 Seasons
类型来实例化 Seasons 值是完全合法的:
Seasons season = (Seasons)5;
更新:首先要注意的是 - System.Enum 是一种值类型(它继承自 ValueType),如 DateTime、Int32 或 Boolean。如果您查看为 Seasons 枚举定义生成的 IL,您会看到
.class public auto ansi sealed Seasons
extends [mscorlib]System.Enum
{
.field public static literal valuetype Foo.Seasons Autumn = int32(2)
.field public static literal valuetype Foo.Seasons Spring = int32(0)
.field public static literal valuetype Foo.Seasons Summer = int32(1)
.field public specialname rtspecialname int32 value__
.field public static literal valuetype Foo.Seasons Winter = int32(3)
}
所以,这是一组公共(public)静态字段,它们被标记为文字 - 这使得编译器将枚举值直接嵌入到代码中。例如。以下代码
Seasons season = Seasons.Autumn;
Console.WriteLine(season);
编译后将只有枚举 Autumn 成员的值:
.locals init (
[0] valuetype Foo.Seasons seasons) // it's value type!
L_0000: nop
L_0001: ldc.i4.2 // here simple integer value 2 is loaded
L_0002: stloc.0
L_0003: ldloc.0
L_0004: box Foo.Seasons
L_0009: call void [mscorlib]System.Console::WriteLine(object)
因此,实际上您已将基础类型的值分配给枚举类型的变量。名称或值没有特殊属性。只是(任何)整数常量。
现在关于获取枚举的所有值 - 当您调用 Enum.GetValues
时,它会返回来自私有(private) Hashtable
fieldInfoHash
的值,您可以在枚举
类。 Hashtable
缓存枚举的名称和值。因此,当您第一次获取枚举值时,HashEntry
具有枚举类型的值(ulong
数组)和名称(string
数组)是创建。实际上,如果您调用其他 Enum
方法,如 GetNames
或 IdDefined
,将使用此缓存。
在为枚举类型缓存条目后,返回它的值数组。但是用一招。这不是 ulong
值的简单数组。每个值都作为枚举类型用 Enum.ToObject
调用装箱(您可以在 System.RuntimeType 中找到实现):
ulong[] values = Enum.InternalGetValues(this);
Array array = Array.UnsafeCreateInstance(this, values.Length);
for (int i = 0; i < values.Length; i++)
{
object obj2 = Enum.ToObject(this, values[i]);
array.SetValue(obj2, i);
}
return array;
这就是为什么每个值对象都是 Seasons 类型而不是 ulong
类型的原因。
关于c# - Enum.GetValues() 的内部工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21183119/
我不明白为什么这甚至可以编译。我尝试过不同的格式,它们似乎都有效。 为什么有一个 enum of enum of enum of.. 是合法的? interface I { enum E im
我有大型数据库(数百万行),我正在尝试为 2 个字段的数据类型做出最佳选择。我制作的大部分内容都是 varchar 或 INT。但是,我想知道 Enum 是否是最好的方法的 2 个字段。 字段 1第一
我是 C++ 新手,我想知道是否有人可以帮助我理解原因 enum difficulty { NOVICE, EASY, NORMAL, HARD, UNBEATABLE }; difficulty m
从 native 代码强制转换(在 C++/CLI 中)的正确方法是什么 enum到托管代码enum其中包含相同的 enum值(value)观?与使用 C# 的强制转换方式有什么区别,例如 (int)
我有以下枚举 [Flags] public enum WeekDays { Monday = 1, Tuesday = 2, Wednesday = 4, Thursd
我刚刚学习 Java 中的枚举。当我运行下面的代码时,我收到一个错误,我也在下面重现该错误。基本上,我的问题是:当我在枚举中定义一个方法,并且在该方法中我想检查枚举的值以便我可以根据该值执行某些操作时
我想要一些语法糖来打开 Enum .当然,一个if else块按预期工作: @enum Fruit apple=1 orange=2 kiwi=3 function talk1(fruit::Frui
我试图在 Enum.each 的函数内为变量设置一个值,但在循环结束时,变量为空,我不知道为什么会出现这种行为。 代码: base = "master" candidates = ["stream",
This question already has answers here: Is there a way to use existing structs as enum variants? (2个
我想让 Java 中的枚举器具有其他枚举作为属性。 public enum Direction { Up(Down), Down(Up), Left(Right), R
我有一个包含两种不同可能“类型”的枚举和一个可以返回其中任何一种的函数,封装在 Option 中: enum Possibilities { First(i32), Second(St
我刚刚注意到 pyhton 中 Enum+defaultdict 的一个非常奇怪的行为。我定义了一个这样的枚举,它收集了一些默认词典: from enum import Enum from colle
我想在运行时从配置文件生成一些类型。为简单起见,假设我已经将数据加载为 Python 字典: color_values = dict(RED = 1, YELLOW = 2, GREEN = 3) 我
我想创建一个方法,在传入参数的任何枚举类上实现 valueOf(并包装一些专门的错误/缺失名称代码)。所以基本上,我有几个枚举,例如: enum Enum1{ A, B, C } enum Enum2
我有一个user数据库表: CREATE TABLE IF NOT EXISTS `user` ( `user_id` INT UNSIGNED NOT NULL AUTO_INCREMENT,
这是我的 JPA @Repository,在这里我们可以得到 list与 findByFullNameContaining(String query) - 通过在查询中提供 fullName 的子字符
(也在这里打开了一个问题:https://github.com/square/moshi/issues/768 但被要求打开一个stackoverflow问题)我正在编写一个通用适配器来转换带有枚举值
请帮助找到以下情况的最佳方法: 有一个表,大约有 20 列。 每列都有自己的短名称、全名称和类型(数字或字符串)。 每个列类型都可以有自己的运算符 - 例如,字符串 - 包含、等于;数字 - 更多、更
如果我在 python 中按功能创建了 enum.Enum(本例中为 3.7),如何从中检索类的名称? import enum def print_the_enum_class_name(some_e
我正在编写一个与第 3 方程序交互的程序。这个第 3 方程序允许用户制作可以运行在第 3 方程序中进行的步骤记录的按钮。 但! 这些按钮还可以运行用户定义的批处理文件。因此,我使用此功能通过创建文件并
我是一名优秀的程序员,十分优秀!