目录
4. 总结
1. enum常用表现形式
什么是枚举类型?
答:如果一个变量只有几种可能的值,那么就可以定义为枚举类型,比如:性别只有男和女,那么就可以将性别定义为一种枚举类型,其中男和女就是性别所包含的变量。所谓”枚举”是指将变量的值一一列举出来,变量的值只能在列举出来的值的范围内。在C++中,枚举类型分为不限定作用域(enum)和限定作用域(enum class)。
enum my_enum
{
my_enum_1 = 0,
my_enum_2,
};
enum
{
my_enum_1 = 0,
my_enum_2,
}
typedef enum
{
my_enum_1 = 0,
my_enum_2,
}my_enum;
2. 常见问题
2.1 问题一:不可重复定义
传统enum关键字的作用域是全局的,如果enum A中声明的一个枚举类型my_enum3,无法在enum B中声明同样的枚举类型,会出现编译错误(会报 重定义错误)。
enum A
{
my_enum3 = 0,
};
enum B
{
my_enum3 = 0,
};
2.2 问题二:隐式转换类型不确定
enum my_enum
{
my_enum1 = 0,
my_enum2,
};
int my_int = my_enum1;
这就导致无法指定底层的数据类型,也就是说,无法明确知道一个枚举类型占用内存的字节数,如果这样的话在结构体中就会出现问题,特别是结构体需要内存对齐或者填充处理的时候。
3. 解决办法enum class和enum struct
enum class 、enum struct是C++11标准——强枚举类,目的是限定作用域,为了解决enum存在的问题。
在c++11标准中,除了传统的enum关键字之外, 还新增了一个概念: enum class, enum struct组合的形式(两者是等价的),当然单纯的enum关键字和enum class组合并不冲突,都能使用。 这一组合的出现就是为了解决传统enum关键字面临的问题。
enum class组合具有class封装性的特性,作用域是确定的。因此基于上述结构特性,下述声明和定义枚举变量是合法的。
enum class A
{
my_enum3 = 0,
};
enum class B
{
my_enum3 = 0,
};
如果要访问A和B中的枚举就需要加上作用域:
A a;
B b;
a = A::my_enum3;
b = B::my_enum3;
3.1 enum class可指定底层的数据类型
3.1.1 在定义枚举时定义枚举类型。
两者处于不同作用域下,不会重复定义。
enum class A:int //每个枚举类型都是int类型的
{
my_enum3 = 0;
}
enum class B:unsigned char //每个枚举类型都是int类型的
{
my_enum3 = 0;
}
int main(int argc, char *argv[])
{
A a = A::my_enum3;
B b = B::my_enum3;
}
3.1.2 在定义枚举时没有定义枚举类型
enum class A
{
my_enum3 = 0;
}
int main(int argc, char *argv[])
{
A a = A::my_enum3;
int aa == a;//错误,无法从“my_enum”转换为“int”
int aa == int(a);//正确,显示将enum class转换为整数
return 0;
}
3.2 枚举成员不能和类中成员函数重名
编译报错:
class Student
{
public:
enum Status
{
boy = 1;
girl = 2;
}
public:
Student();
~Student();
void learn();
}
编译通过:
class Student
{
public:
enum class Status
{
boy = 1;
girl = 2;
}
public:
Student();
~Student();
void learn();
}
这便体现C++11引入枚举类(enum class)的重要性,enum class能够有效对枚举作用域进行限定,避免了枚举成员的重定义和枚举成员名称和外部类成员函数名称的重定义。
4. 总结
enum class, enum struct组合的出现可以极大的增加枚举类型使用的灵活性,安全性以及易用性。
注意事项:
【1】枚举中的成员结束符是“,”;
【2】枚举初始化可以赋值负数,以后标识符依次加1;
【3】在外面可以对枚举变量进行赋值,但是需要进行类型转换;
【4】枚举定义被限制在枚举作用域内,并且不能隐式转换为整数类型,但是可以显式转化为整数类型。