C++语言导学 第二章 用户自定义类型 - 2.5 枚举
2.5 枚举
除了类之外,C++还提供了一种形式简单的用户自定义类型,可以用来枚举一系列值:
enum class Color{red, blue, green};
enum class Traffic_light{green, yellow, red};
Color col = Color::red;
Traffic_light light = Traffic_light::red;
注意,枚举值(如red)位于其enum class的作用域之内,因此我们可以在不同的enum class中重复使用这些枚举值而不致引起混淆。例如,Color::red是指Color的red,它与Traffic_light::red显然不同。
枚举类型常用于描述规模较小的整数值集合。通过使用有指代意义的(且易于记忆的)枚举值名字,可以提高代码的可读性,降低出错的风险。
enum后面的class关键字指明了枚举是强类型的,且它的枚举值位于指定的作用域中。不同的enum class是不同的类型,这有助于防止对常量的意外误用。例如,我们不能混用Traffic_light和Color的值:
Color x = red; //错误:哪个red?
Color y = Traffic_light::red; //错误:这个red不是一个Color
Color z = Color::red; //正确
同样,我们也不能隐式地混用Color和整数值:
int i = Color::red; //错误:Color::red不是一个int
Color c = 2; //初始化错误:2不是一个Color
捕捉试图向枚举类型的转换是避免错误的一种好的防御措施,但我们常常希望用枚举类型的基础类型(默认是int)的值对其初始化,这就是允许从基础类型隐式转换为枚举类型:
Color x = Color{5}; //正确,但有些啰嗦
Color y{6}; //也是正确的
默认情况下,enum class只定义了赋值、初始化和比较(如==和<)操作。然而,既然枚举类型是一种用户自定义类型,那么就可以为它定义别的运算符:
Traffic_light& operator++(Traffic_light& t) //前置递增运算符++
{
switch(t){
case Traffic_light::green: return t = Traffic_light::yellow;
case Traffic_light::yellow: return t = Traffic_light::red;
case Traffic_light::red: return t = Traffic_light::green;
}
}
Traffic_light next = ++light; //next变成了Traffic_light::green
如果你不想显式地限定枚举值名字,并且希望枚举值可以是int(无须显式转换),你可以去掉enum class中的class而得到一个“普通”enum。“普通”enum中的枚举值的作用域与其enum的作用域一致,并且会隐式地转换成整数值。例如:
enum Color{red, green, blue};
int col = green;
在这里,col的值是1.默认情况下,枚举值对应的整数从0开始,依次加1。“普通”enum很早就出现在C++和C中了,所以即使它的效果并不是那么好,在当前的代码中仍很常见。