C++11新特性——strongly-typed enums

        <link rel="stylesheet" href="https://csdnimg.cn/release/phoenix/template/css/ck_htmledit_views-f57960eb32.css">
                          <link rel="stylesheet" href="https://csdnimg.cn/release/phoenix/template/css/ck_htmledit_views-f57960eb32.css">
      <div class="htmledit_views" id="content_views">

一.传统枚举类型的缺点

1.作用域冲突

传统C++中枚举元素被暴漏在外层作用域中,这样若是同一作用域下有两个不同的枚举类型,但含有相同的枚举元素,就会产生冲突。

  1. enum EnumOne
  2. {
  3. A = 1,
  4. B = 2
  5. };
  6. enum EnumTwo
  7. {
  8. A = 1,
  9. C = 2
  10. };

error C2365: “A”: 重定义;以前的定义是“枚举数”

2.无关的枚举元素可以比较

另外一个缺陷是传统枚举类型的枚举元素总是被隐式转换为整形,这就使得毫无关系的两个枚举类型的枚举元素可以进行比较。
  1. enum EnumOne
  2. {
  3. A = 1,
  4. B = 2
  5. };
  6. enum EnumTwo
  7. {
  8. C = 1,
  9. D = 2
  10. };
  11. int main()
  12. {
  13. if (EnumOne::A == EnumTwo::C)// 直接写AC也行
  14. {
  15. cout << "equals" << endl;// 输出equals
  16. }
  17. if (EnumOne::A == 1)
  18. {
  19. cout << "equals too" << endl;// 输出equals too
  20. }
  21. system("pause");
  22. return 0;
  23. }

二.强类型枚举

强类型枚举(Strongly-typed enums),号称枚举类型,是C++11中的新语法,用以解决传统枚举类型存在的缺点。

它不会将枚举元素暴露到外层作用域中,也不会隐式转换为整形,并且可以拥有用户指定的元素类型(传统枚举也增加了这个性质)

强类型枚举使用enum class语法来声明,如下:
  1. enum class EnumOne
  2. {
  3. A = 1,
  4. B = 2
  5. };
  6. enum class EnumTwo
  7. {
  8. A = 1,
  9. C = 2
  10. };

此时这两个枚举即使有同名元素A,编译器不会报错了。

但是这两个枚举中枚举元素的比较,或者枚举元素与整数的比较都会报错。例如:
  1. enum class EnumOne
  2. {
  3. A = 1,
  4. B = 2
  5. };
  6. enum class EnumTwo
  7. {
  8. A = 1,
  9. D = 2
  10. };
  11. int main()
  12. {
  13. if (EnumOne::B == EnumTwo:😄)// A和B必须带上作用域,不能直接写成AB,此时单独的A或B不再有意义
  14. {
  15. cout << "equals" << endl;// 输出equals
  16. }
  17. if (EnumOne::B == 2)
  18. {
  19. cout << "equals too" << endl;// 输出equals too
  20. }
  21. system("pause");
  22. return 0;
  23. }

两处相等判断都在编译时报错:error C2678: 二进制“==”: 没有找到接受“EnumOne”类型的左操作数的运算符(或没有可接受的转换)
三.比较
传统枚举类型和强枚举类型都支持用户指定元素类型(默认为int类型)
  1. enum EnumA : int { A,B};
  2. enum class EnumB : long {C,D};

还有一点值得说明的是C++11中枚举类型的前置声明也是可行的,比如:
  1. enum Enum1; // 不合法
  2. enum Enum2 : unsigned int; // 合法的 C++11
  3. enum class Enum3; // 合法的 C++11,默认为int
  4. enum class Enum4 : unsigned int;// 合法的 C++11
  5. enum Enum2 : unsigned short; // 不合法的 C++11,Enum2已被声明为unsigned int


  1. enmu class Clolor:char; //前置声明枚举
  2. void Foo(Color*p); //前置声明的使用
  3. //....................
  4. enum class Color:char{RED,GREEN,BLACK,WHITE}; //前置声明的定义

四.一个例子

  1. #include <iostream>
  2. using namespace std;
  3. int main()
  4. {
  5. enum class Status {Ok, Error};
  6. enum struct Status2{Ok, Error};
  7. Status flag = Status::Ok;
  8. if(flag == Status::Ok)
  9. {
  10. cout << "equals" <<endl;//equals
  11. }
  12. //int n=flag; // 错误,不会隐式转换为int
  13. int n = static_cast<int>(flag); // 正确, n = 0,枚举元素默认从0开始
  14. enum class EnumOne : char { A = 1, B = 2}; //指定枚举的底层数据类型
  15. enum class EnumTwo : unsigned int { C = 1, D = 2, Dbig = 0xFFFFFFF0U };
  16. cout << sizeof(EnumOne::A) << endl; // 1
  17. cout << (unsigned int)EnumTwo::Dbig << endl; // 编译器输出一致,4294967280
  18. cout << sizeof(EnumTwo::D) << endl; // 4
  19. cout << sizeof(EnumTwo::Dbig) << endl; // 4
  20. system("pause");
  21. return 0;
  22. }



posted @ 2019-05-19 21:45  unique_ptr  阅读(458)  评论(0编辑  收藏  举报