为什么枚举在 C 中作为一种类型存在

Stu*_*ent 2 c enums

我从教科书中了解到,枚举的典型定义如下:

enum weather {
    sunny,
    windy,
    cloudy,
    rain,
} weather_outside;
Run Code Online (Sandbox Code Playgroud)

然后声明一个 var,如下所示:

enum weather weather_outside = rain;
Run Code Online (Sandbox Code Playgroud)

我的问题是,如果可以通过说eg来使用枚举常量rain,它保​​持整数3,那么具有类似类型的更复杂的减速等于3的确切用途和意义是什么enum weather weather_outside = rain;weather_outside因为枚举值只能是编译时常量)?为什么不直接使用 aconst或 宏呢?我有点困惑枚举是否真的有必要?!

Eri*_*hil 5

C 中的枚举主要是为了方便,因为它们不是强类型的。它们是为了表达命名选项而创建的,但语言开发的限制以及人们将它们用于其他用途的方式导致了目前的情况,它们只不过是命名的整数值。

\n

枚举支持我们有各种不同选项的情况,例如您显示的天气条件,并希望为它们命名。理想情况下,枚举应该是强类型的,这样rain就不容易转换为 3,反之亦然;编写int x = rain;enum weather x = 3;都会从编译器中产生警告或错误。

\n

然而,这样做存在一些问题。考虑一下我们何时想要编写处理枚举中所有值的代码,例如:

\n
for (enum weather i = sunny; i <= rain; i = i+1)\n    DoSomethingWithWeatherCondition(i);\n
Run Code Online (Sandbox Code Playgroud)\n

看一下更新表达式,i = i+1。这对于经验丰富的 C 程序员来说是非常自然的。(我们可以写i++,但这是同一件事,这里将其拼写出来以供说明。)我们知道它会更新i为下一个值。但是,当我们思考一下,什么是i+1i是一个枚举值,并且1是一个int. 所以我们添加了两个不同的东西。

\n

为了实现这一点,C 将枚举值视为整数。这允许i+1以普通方式计算两个整数相加。此外,结果是 an int,并且我们有,这意味着我们必须允许将 an 分配给 an 。i = some int resultintenum weather

\n

也许解决这个问题的一种方法是定义枚举值和整数的加法,这样就i+1不需要将其视为i整数;它只是被指定返回枚举中的下一个值 after i。但早期的C开发并没有做到这一点。这将需要更多的工作,并且编程语言中的新功能并不是在预见到哪些功能有用或可能出现哪些问题的情况下一次性全部开发出来的。它们通常是一点一点开发的,用一些原型代码尝试新事物。

\n

因此,枚举值是整数。一旦它们成为整数,人们就开始将它们用于超出简单原始目的的用途。枚举对于定义可在编译器需要常量表达式的地方使用的常量非常有用,包括数组维度和静态对象的初始值。const当时不存在,但它不会起作用,因为定义了const int x = 3;static const int x = 3;,我们无法x在 中使用它float array[x];。(当时还不存在变长数组,即使现在它们也不适用于静态对象。)当 的定义在函数外部时我们也不能使用xin (因此它定义了静态对象)。但是,如果被定义为枚举值而不是int m = 2*x+3;mxint,则它可以用于这些目的。

\n

这导致在没有真正枚举事物的情况下使用枚举。例如,它们通常用于各种位掩码:

\n
enum\n{\n    DeviceIsReadable           = 1,\n    DeviceIsWriteable          = 2,\n    DeviceSupportsRandomAccess = 4,\n    DeviceHasFeatureX          = 8,\n    \xe2\x80\xa6\n}\n
Run Code Online (Sandbox Code Playgroud)\n

一旦人们开始以这种方式使用枚举,再将枚举设为强类型并在其上定义算术就为时已晚了。这些位掩码必须可与按位运算符|, &, and 一起使用^,而不仅仅是+1. 人们将它们用于任意常数和算术运算。重新定义 C 语言的这一部分并更改现有代码太困难了。

\n

因此,枚举在 C 中从未被开发为正确的独立类型。

\n