在时间::持续时间类型之间转换,其中两个分母的LCM都很大
Converting between chrono::duration types where the LCM of both denominators is large
不久前我遇到了一个问题,我必须在两种不同的持续时间类型之间转换,如下所示:
using MyType_T = duration<long long, ratio<1, 4294967296LL>>;
using OtherType_T = duration<long long, ratio<1, 10000000>>;
当我使用::boost::chrono::duration_cast<>时(或者std::chrono::duration_cast<>),从MyType_T到OtherType_T,反之亦然,我发现当源值超过某个限制时,这些类型的转换可能会产生错误的结果。
仅调用duration_cast(tm)存在问题的原因是,两个分母的LCM完全用56位表示。在这种情况下,转换是通过将tm中的值除以分母的LCM来完成的,并将其作为公共因子类型传递到OtherType_T持续时间的构造中。这将执行乘法运算以获得正确的值。
正如你所看到的,对于足够大的值(在这种情况下为64或更大),这意味着除法可以将所有值移到零,而你什么都得不到。
除了执行以下操作外,还有什么方法可以在不损失所有精度的情况下进行转换吗?
- 创建一个新类型Interim_T,它只是两者中的common_type,去掉了分母。
typedef typename common_type<MyType_T, OtherType_T>::type CommonDuration;
- 对新类型的"From"值tm执行duration_cast,以获得一些值高位。
auto highbits = duration_cast<duration<CommonDuration::rep, ratio<CommonDuration::num>>>(tm);
- 从原始值tm中减去高位,以获得值低位。
auto lowbits = tm - highbits;
- 对于高位,添加duration_cast的结果(低位)。这就是结果。
auto result = highbits + duration_cast<OtherType_T>(lowbits);
我会使用浮点类型作为临时类型:
using MyType_T = duration<long long, ratio<1, 4294967296LL>>;
using OtherType_T = duration<long long, ratio<1, 10000000>>;
using OtherType_F = duration<long double, OtherType_T::period>;
MyType_T x{0x7FFFFFFFFFFF};
auto y = duration_cast<OtherType_T>(OtherType_F{x});
语法更直观,逻辑也不那么复杂。请注意,如果结果不能容纳在OtherType_T
中,则仍然会溢出。
相关文章:
- 如何在C++中从两个单独的for循环中添加两个数组
- 为什么两个不同的未命名名称空间可以共存于一个cpp文件中
- 当在同一名称空间中有两个具有相同签名的函数时,会发生什么
- 如何返回一个类的两个对象相加的结果
- 如何在C++中将一个无符号的 int 转换为两个无符号的短裤?
- 如何将两个不同矢量的同一位置的两个元素组合在一起
- 两个字符串在 c++ 中不相等
- 在两个类中共享相同的函数调用,并在不需要时避免空实例化
- 两个文件使用彼此的功能-如何解决
- 为什么Mat类的两个对象可以在不重载运算符+的情况下添加
- 如何确保在使用基于布尔值的两个方法之一调用方法时避免分支预测错误
- 停止cmake target_link_libraries将插件中静态库的两个对象文件链接到静态库本身
- 将fold表达式与std::一起用于两个元组
- 如何在C++中比较两个char数组
- 给定两个偶数,求出它们之间所有偶数的平方和
- 比较两个大小不等的映射c++
- C++需要帮助从用户那里获得一个整数,并确保它在另外两个整数之间
- 如何在for循环中包含两个索引值的测试条件
- 在声明中合并两个常量"std::set"(不是在运行时)
- 在时间::持续时间类型之间转换,其中两个分母的LCM都很大