条款十 倾向使用范围enum

简介

一般来说,在大括号内定声明的名字只在括号内可见,但是C++98的枚举是个例外,名字属于包含enum关键字的作用域。

// this kind of enum: unscoped
enum Color {black, white, red};
auto white = false; // error

C++14enum则改变了这种情况:

enum class Color {black, white, red};
auto white = false; // fine
Color c = white; // error
Color c = Color::white; // fine
auto c = Color::white; // fine

由于范围内的枚举使用enum class,所以有时也被称作枚举类。

此外,范围enum是强类型的,非范围enum则会隐式转换为整型。

// 正常的代码
enum Color {black, white, red};
std::vector<std::size_t> primeFactors(std::size_t x);
Color c = red;
if(c < 14.5) {
    auto factors = primeFactors(c);
}

在添加class后,上述代码出现错误,除非使用cast显式转换。

// 正常的代码
enum class Color {black, white, red};
std::vector<std::size_t> primeFactors(std::size_t x);
Color c = Color::red;
if(static_cast<double>(c) < 14.5) {
    auto factors = primeFactors(static_cast<std::size_t>(c));
}

此外,范围enum可以提前声明,即可以先声明枚举类,之后再指出具体枚举值。

当处理std::tuple的时候,非范围enum则十分有用。

总结

  • C++98风格的enum如今被称为非范围enum

  • 范围enum的枚举值只在enum内可见,且只能通过cast转换为其他类型

  • 范围和非范围enum都支持指定底层类型。范围enum的默认底层类型是int,而费范围没有

  • 范围enum可以提前声明。非范围类型只有在他们的声明指定底层类型的时候才可以提前声明


Azathoth
26 声望1 粉丝