学科分类
目录
C++基础

限定作用域的枚举类型

枚举类型是将一组有限的整数常量组织在一起用以描述变量可取值范围的一种数据类型。C++中有两种类型的枚举:不限定作用域的枚举类型和限定作用域的枚举类型。限定作用域的枚举类型是C++11标准引入的新类型。

● 限定作用域枚举类型是为了弥补不限定作用域枚举类型的不足而出现的,不限定作用域的枚举类型不是类型安全的,主要表现在如下几个方面:

● 不限定作用域的枚举类型中的枚举成员被视为整数,两种不同的枚举类型之间可以进行比较。两种不同类型的数据进行比较,可能带来数据类型转换,引起数据表示不完整。

● 不限定作用域枚举所使用的整数类型及其大小都由实现方法定义,皆无法明确指定。

● 不限定作用域枚举类型的枚举成员与枚举类型外部数据处在同一个作用域范围内,多个枚举类型不能有同名的枚举成员。

C++11标准引入的限定作用域的枚举类型,其定义方式如下所示:

enum class 枚举类型名称 {枚举成员1, 枚举成员2, …, 枚举成员n};

相对于不限定作用域的枚举类型,限定作用域的枚举类型定义的枚举成员在枚举类型作用域外不可访问。

//定义不限定作用域枚举类型
enum color {red, yellow, green};  
//错误,不限定作用域枚举成员不能与其它枚举类型成员同名
enum stoplight{red, yellow, green}; 

//正确,限定作用域枚举成员的作用域限定在类型内
enum class newcolor {red, yellow, green}; 
//正确,不限定作用域的枚举类型外可以使用枚举成员
color e_color = green;       

//错误,限定作用域的枚举成员在类型外不可访问
//虽然color类型的green成员可访问,但是color与newcolor类型不同,不可赋值
newcolor ec_color = green;   

在旧标准中,枚举变量可以使用整数常量进行赋值,但在C++11标准中,要想初始化枚举类型对象或对枚举对象赋值,必须使用该类型的一个枚举成员或该类型的一个对象。示例代码如下所示:

enum color {red, yellow, green};  
enum class newcolor {red, yellow, green};  

color e_var1 = 1;                 //错误,1不是枚举类型值
color e_var2 = red;                //正确,red是color类型的枚举值
color e_var3 = e_var2;             //正确,同类型数据之间的赋值

newcolor e_nvar1 = newcolor::red;       //正确,red是newcolor类型的枚举值

不限定作用域的枚举类型对象或枚举成员可以自动转化为整型,因此可以在需要整数的地方使用它们,限定作用域的枚举类型却没有此特点。示例代码如下所示:

int i = color::red;               //正确,不限定作用域的枚举成员可转化为整数
int j = newcolor::red;             //错误,限定作用域的枚举成员不可转化为整数

C++11新标准除了增加了限定作用域枚举类型外,还改进了原有的不限定作用域枚举类型的定义形式:

1、枚举类型中枚举成员的类型可以显式指定,指定方式是在枚举类型名称后加冒号和枚举成员类型。

enum intvalues:unsigned long long{
  chartype = 255, shorttype = 65535, inttype = 4295967295,
  longtype = 4294967295ul,
  longlongtype = 18446744073709551615ul
};

若不指定枚举成员类型,限定作用域枚举类型成员默认为int,不限定作用域枚举类型成员不存在默认类型,能够确定的是成员类型足够大到能容纳所有成员。指定了枚举成员类型后,若某一枚举成员值超出了该类型所能容纳的范围,将引发错误。

2、在C++11 中,可以提前声明enum。enum的前置声明必须指定枚举成员类型,隐式指定或显式指定均可。

enum etype1;              //错误,不限定作用域枚举类型的前置声明应指定成员类型
enum etype2:unsigned int;      //正确
enum class etype3;          //正确,限定作用域枚举类型成员有默认类型int 
enum class etype4:unsigned int;  //正确,前置声明指定枚举成员类型

与所有的声明语句一样,枚举的声明和定义中描述的成员类型必须一致,另外不能在同一个上下文中先声明一个不限定作用域的枚举类型再声明一个同名的限定作用域枚举类型。

enum class type1;
//错误,不能定义同名的不限定作用域枚举类型和限定作用域枚举类型
enum type1;       
点击此处
隐藏目录