在C++中与枚举相乘
Multiplying an Enum in C++
我有一些代码将枚举与整数相乘:
QuantLib::Date date2 = date + 12 * QuantLib::Months;
其中QuantLib::Months定义为:
enum TimeUnit { Days,
Weeks,
Months,
Years
};
这给了我所需的结果,即日期2与时间相隔一年。然而,我无法理解这是如何实现的。
我原以为这不会编译。现在我觉得我得到了一个"12months"对象,然后由QuantLib::Date"+"运算符重载处理,但我以前从未见过这种风格。
我有C#背景,所以在这里的工作中可能有一些我不知道的事情。有人能解释一下发生了什么吗?如有任何参考文件,不胜感激。
以下其中一项在此处有效:
-
在C++中,枚举类型可以隐式转换为整型。如果在这里发生这种情况,
date + 12 * QuantLib::Months
将与date + 12 * 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
现在将是将来now
的time_point
24 * 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类型的对象中。然后,在乘法运算中执行常见的算术转换。
- 不带大括号的枚举形式
- 枚举环境变量的惯用C++14/C++17方法
- 类似枚举的计算常量
- 如何正确实现和访问运算符的各种自定义枚举器
- 错误:从"int"到枚举c++的转换无效
- C++中构造函数中的枚举
- 访问在 C++ 结构中声明的枚举变量
- 枚举类'classname'的多重定义
- 强枚举类型定义:Clang Bug 还是 C++11 标准不确定性?
- typedef 枚举和枚举类有什么区别?
- 为什么我的开关/机箱在使用枚举时默认?
- 标准::可选枚举的比较运算符
- C++两个源文件之间共享的枚举的静态实例
- 打印没有铸件的枚举可以在C++中吗?
- 枚举成员与静态 int 成员?
- C++:枚举:错误:应使用标识符而不是"}"
- 带有 c++ 的枚举(输入检查)
- 在 qml 中使用 Q_ENUM 和 Q_PROPERTY 作为枚举类
- 为什么 int 类型的枚举类值不能用作 int
- 在 C++ 中输出枚举类类型的向量元素