• C++ enum枚举类型详解

    《enum枚举用法攻略》一节已经介绍过枚举数据类型,还记得吗?它们是由程序员定义的数据类型,由一组称为枚举量的值组成,枚举量代表整数常量。本节将进一步介绍枚举数据类型的应用,以及使用它们能做和不能做的事情。

    在同一个语句中声明 enum 数据类型并定义变量

    以下代码使用了两行来声明一个枚举数据类型,并定义了该类型的变量:

    enum Car {PORSCHE, FERRARI, JAGUAR};
    Car sportsCar;

    但是,C++ 允许在同一个语句中声明一个枚举数据类型,并定义该类型的一个或多个变量。因此,上面的代码可以改写为如下形式:

    enum Car {PORSCHE, FERRARI, JAGUAR} sportsCar;

    以下语句不但声明了 Car 数据类型,而且定义了 2 个变量 myCar 和 yourCar:

    enum Car {PORSCHE, FERRARI, JAGUAR} myCar, yourCar;

    将整数赋值给 enum 变量

    虽然枚举数据类型的枚举量在内存中是以整数形式存储的,但是,并不能直接将整数值赋给 enum 变量。例如,假设程序中包含以下声明:

    enum Day { MONDAY, TUESDAY, WEDNESDAY, THURSDAY, FRIDAY };
    Day today;

    现在可以编写以下赋值语句:

    today = THURSDAY;

    但是,正如前面已经介绍过的,以下语句是非法的。如果试图编译它,则会出现一个错误消息。

    today = 3; //错误

    在给 enum 变量赋值时,应该使用有效的枚举量。但是,如果某些情形要求必须将整数值存储在 enum 变量中,则可以通过强制转换的方式,将整数转换为 enum 数据类型,示例如下:

    today = static_cast<Day>(3);

    该语句的作用与以下语句是一样的:

    today = THURSDAY;

    将枚举量赋值给 int 变量

    虽然不能直接将整数赋值给 enum 变量,但是,反过来却是可以的,即可以将枚举量赋值给整型变量。例如,以下代码将可以正常运行:

    enum Day { MONDAY, TUESDAY, WEDNESDAY, THURSDAY, FRIDAY };
    int today = THURSDAY;
    Day workday = FRIDAY;
    int tomorrow = workday;
    cout << today <<" "<< tomorrow << endl;

    当该代码运行时,它将显示 3 4

    使用数学运算符改变 enum 变量的值

    虽然枚举量实际上就是整数,而且 enum 变量真正保存的也是整数值,但是,如果试图使用它们执行数学运算,那么还是会遇到问题。

    例如,来看以下代码:

    Day day1, day2; // 定义 2 个 Day 变量
    day1 = TUESDAY; // 将 TUESDAY 赋值给 day1
    day2 = day1 + 1; //错误!该语句不起作用

    第 3 个语句之所以会出现问题,是因为表达式 day1+1 的结果为整数值 2,然后赋值运算符会试图将整数值 2 赋给 enum 变量 day2,但是前面已经介绍过,C++ 并不会隐式地将 int 转换为 Day 类型,所以出现了问题。

    要解决该问题,可以如前文所述,通过强制转换的方式,将整数值结果转换为 Day 数据类型,示例如下:

    day2 = static_cast<Day> (day1 + 1); //该语句有效

    使用枚举量输出值

    正如前文所述,将枚举量发送到 cout 将显示枚举量的整数值。例如,假设使用前面定义的 Day 类型,以下语句将显示 0:

    cout << MONDAY << endl;

    如果要让枚举量显示诸如“Monday”这样的字符串,则需要编写代码来产生所需的字 符串。例如,如果 workDay 是一个已经被初始化为某个值的 Day 变量,则以下switch语句将根据变量的值显示对应的日期名称:

    switch(workDay)
    {
        case MONDAY :
            cout <<"Monday";
            break;
        case TUESDAY :
            cout <<"Tuesday";
            break;
        case WEDNESDAY:
            cout <<"Wednesday";
            break;
        case THURSDAY :
            cout <<"Thursday";
            break;
        case FRIDAY :
            cout <<"Friday";
            break;
    }

    使用枚举量控制循环

    因为枚举量是作为整数存储在内存中的,所以可以使用它们控制循环迭代的次数。但是,如前文所述,不能直接将数学运算的结果赋值给枚举量,而必须先将结果强制转换为 enum 数据类型。所以,不能直接对 enum 变量使用 ++ 或 -- 运算符。以下语句将无法运行。

    Double sale, total = 0.0;
    for ( Day workday = MONDAY; workday <= FRIDAY ; workday++) // 错误
    {
        cout << "Enter the sales for day " << (workday +1) << ": ";
        cin >> sales;
        total += sales;
    }

    要解决该问题,可以将以上语句修改为如下形式:

    for ( Day workday = MONDAY; workday <= FRIDAY; workday = static_cast<Day>(workday + 1))

    但是,更简单的方法是将循环控制变量变成 int 类型。按照这个思路,可以将上面的语句修改如下:

    for ( int workday = MONDAY; workday <= FRIDAY;workday++)
    {
        cout << "Enter the sales for day " << (workday + 1) <<":";
        cin >> sales;
        total += sales;
    }

    使用C++ 11中的强类型 enum

    C++ 不允许相同作用域内的多个枚举量具有相同的名称,也就是说,在同一作用域内,即使是两个不同的枚举数据类型,也不能定义或使用相同的枚举量名称作为自己的成员。但是,C++11 包含了一个新类型:enum,即所谓的强类型枚举,也称为 enum 类,可以摆脱这种限制。

    以下是强类型 enum 声明的 2 个示例:

    enum class Presidents { MCKINLEY, ROOSEVELT, TAFT };
    enum class VicePresidents { ROOSEVELT, FAIRBANKS, SHERMAN };

    以上语句定义了 2 个强类型 enum:Presidents 和 VicePresidents。注意,它们看起来和常规 enum 声明是一样的,区别在于,在 enum 后面多了一个单词 class。虽然这 2 个 enum 包含了相同的枚举量 ROOSEVELT,但是它们在编译时已经不会出错了。

    但是,在使用强类型枚举数据类型时,必须给每个枚举量添加它所属 enum 名作为前缀,后接 :: 操作符。以下是 3 个示例:

    Presidents prez = Presidents::ROOSEVELT;
    VicePresidents vp1 = VicePresidents::ROOSEVELT;
    VicePresidents vp2 = VicePresidents::SHERMAN;

    第一个语句定义了名为 prez 的 Presidents 变量,并且使用了 Presidents::ROOSEVELT 枚举量作为它的初始化值;第二个语句定义了名为 vpl 的 VicePresidents 变量,并且使用了 VicePresidents::ROOSEVELT 枚举量作为它的初始化值;第三个语句定义了名为 vp2 的 VicePresidents 变量,并且使用了 VicePresidents::SHERMAN 枚举量作为它的初始化值。

    请注意,即使枚举量 SHERMAN 是枚举数据类型中唯一的成员,也必须加上它所属的 enum 名称作为前缀。

    以下是使用枚举量比较 prez 变量的if语句示例:

    if (prez == Presidents::ROOSEVELT)
        cout <<"Roosevelt is president! \n";

    强类型枚举量和常规枚举量一样,也是作为整数存储的,但是,如果要检索一个强类型枚举量底层的整数值,则必须使用强制转换运算符。以下示例可以将 Presidents::ROOSEVELT 枚举量底层的整数值赋值给变量 x:

    int x = static_cast<int>(Presidents::ROOSEVELT);

    以下是另外一个示例,它显示了 Presidents::ROOSEVELT 和 Presidents::TAFT 枚举量的整数值:

    cout<<static_cast<int>(Presidents :: ROOSEVELT) << " " <<static_cast<int>(Presidents::TAFT) << endl;

    在声明强类型 enum 时,可以选择性地指定任意整数数据类型作为底层类型。只要在 enum 名称后面加上一个冒号(:),然后跟上所需的数据类型即可。例如,以下语句声明了一个使用 char 数据类型作为其枚举量的 enum:

    enum class Day : char { MONDAY, TUESDAY, WEDNESDAY, THURSDAY, FRIDAY };

    以下语句显示了另外一个示例。该语句声明了一个名为 Water 的 enum,使用了 unsigned 作为其枚举量的数据类型。此外,还给枚举量赋值了。

    enum class Water : unsigned { FREEZING = 32, BOILING = 212 };

    下面的程序演示了强类型枚举数据类型的用法:

    #include <iostream>
    using namespace std;
    
    enum class Presidents { MCKINLEY, ROOSEVELT, TAFT };
    enum class VicePresidents { ROOSEVELT, FAIRBANKS, SHERMAN };
    
    int main()
    {
        Presidents prez = Presidents::ROOSEVELT;
        VicePresidents vp1 = VicePresidents::ROOSEVELT;
        VicePresidents vp2 = VicePresidents::SHERMAN;
        cout << static_cast<int>(prez) << " " << static_cast<int>(vp1) << " " << static_cast<int> (vp2) << endl;
        return 0;
    }

    程序输出结果:

    10 2

更多...

加载中...