将 MsgPack 时间戳 64 字段转换为 DateTime

问题描述

我正在尝试在 C# 中实现 MsgPack TimeStamp 64 type。这是规范所说的:

timestamp 64 stores the number of seconds and nanoseconds that have elapsed since 1970-01-01 00:00:00 UTC
in 32-bit unsigned integers:
+--------+--------+--------+--------+--------+------|-+--------+--------+--------+--------+
|  0xd7  |   -1   | nanosec. in 30-bit unsigned int |   seconds in 34-bit unsigned int    |
+--------+--------+--------+--------+--------+------^-+--------+--------+--------+--------+

* Timestamp 64 format can represent a timestamp in [1970-01-01 00:00:00.000000000 UTC,2514-05-30 01:53:04.000000000 UTC) range.
* In timestamp 64 and timestamp 96 formats,nanoseconds must not be larger than 999999999.

请注意,秒字段是三十位,而不是三十二位。因此,可以通过将 32 位最大值 unsigned int 左移两位并设置现在清除的两个最低有效位来计算最大值。

我尝试使用以下 C# 代码验证这一点,但得到了不同的结果:

DateTime epoch = new DateTime(1970,1,DateTimeKind.Utc); // the zero starting point
long maxSecs = ((long)uint.MaxValue << 2) | 3; // 17179869183
long maxNanoSec = 999999999; // not uint.MaxValue
TimeSpan maxNanoSecSpan = new TimeSpan(maxNanoSec * 100); // 1 tick = 100 nanosec.
DateTime MaxFExt8 = epoch.AddSeconds(maxSecs).Add(maxNanoSecSpan); 
Console.WriteLine(string.Concat("Timestamp 64 : ",epoch.ToString("yyyy-MM-dd HH:mm:ss.fffffff")," - ",MaxFExt8.ToString("yyyy-MM-dd HH:mm:ss.fffffff")));

输出为:

时间戳 64:1970-01-01 00:00:00.0000000 - 2514-05-30 04:39:42.9999900

2 小时 46 分 39 秒太多了...

opening an issue 关于差异之后,其他人检查并发现他们得到了与规格相同的结果。一个使用 MacOS 内置的 NSDate 类型,另一个使用 ruby​​。

我认为我的 6 行代码中一定有一个错误,但我已经盯着它看太久了...

附言请注意,这还不是 MsgPack 实现,在开始使用单元测试验证我的真实代码之前,我首先需要正确地执行此操作。如果基础已经被破坏,小端和大端之间的转换与那些移位的位只会增加更多的混乱......

解决方法

您在除法时乘以纳秒:TimeSpan maxNanoSecSpan = new TimeSpan(maxNanoSec * 100);