在C++中与枚举相乘

Multiplying an Enum in C++

本文关键字:枚举 C++      更新时间:2023-10-16

我有一些代码将枚举与整数相乘:

QuantLib::Date date2 = date + 12 * QuantLib::Months;

其中QuantLib::Months定义为:

enum TimeUnit { Days,
Weeks,
Months,
Years
};

这给了我所需的结果,即日期2时间相隔一年。然而,我无法理解这是如何实现的。

我原以为这不会编译。现在我觉得我得到了一个"12months"对象,然后由QuantLib::Date"+"运算符重载处理,但我以前从未见过这种风格。

我有C#背景,所以在这里的工作中可能有一些我不知道的事情。有人能解释一下发生了什么吗?如有任何参考文件,不胜感激。

以下其中一项在此处有效:

  1. 在C++中,枚举类型可以隐式转换为整型。如果在这里发生这种情况,date + 12 * QuantLib::Months将与date + 12 * 2相同。

  2. 也可以重载枚举类型的运算符。在这种情况下,可能是库定义了一个operator* (int, QuantLib::TimeUnit),它返回与您正在执行的+兼容的内容。

我不知道QuantLib,但我想#2就是正在发生的事情。QuantLib文档证实了这一点(感谢@DaliborFrivaldsky的链接)。

默认情况下,所有枚举基本上都是整数常量,作为所有整数值,您可以在算术表达式中使用它们。

在您的情况下,常量QuantLib::Months的值为2,因为枚举从零开始并简单地增加。


但是,除非您来制作自己的数据/时间功能,否则我建议您使用标准库<chrono>标头中提供的功能(如果您没有支持C++11的编译器/库,则使用Boost chrono)。它内置了所有这些功能。

这里有一个类似于你的代码的例子

auto now = std::chrono::system_clock::now();  // The current time at the moment
auto then = now + std::chrono::hours(24 * 365);

变量then现在将是将来nowtime_point24 * 365小时。

在您的示例中,您使用的是所谓的无范围枚举。每个枚举都有一个底层的积分类型(int类型不是必需的)。C++中没有枚举的默认规则。根据C++标准第4.5.3段(您要求参考文档):

3未分级枚举类型的prvalue,其基础类型为不固定(7.2)可以转换为以下类型可以表示枚举的所有值(即如7.2中所述的bmin到bmax范围内的值):int,无符号int、长整型、无符号长整型、长整型或无符号long-long-int。如果该列表中的任何类型都不能表示枚举的值,非范围枚举类型的prvalue可以转换为具有最低值的扩展整数类型的prvalue整数转换秩(4.13)大于longlongin的秩可以表示枚举的所有值。如果有是两种这样的扩展类型,则选择有符号的类型。

因此,在您的示例中,QuantLib::Months被转换为int,因为枚举的所有值都可以存储在int类型的对象中。然后,在乘法运算中执行常见的算术转换。