为了账号安全,请及时绑定邮箱和手机立即绑定

将 Win32 纪元的纳秒添加到 Go 时间

将 Win32 纪元的纳秒添加到 Go 时间

Go
当年话下 2022-05-10 13:43:12
我正在尝试使用 Go 处理一些 WMI 计数器(作为学习 Go 的一部分),并试图弄清楚如何生成必要的时间对象。基数是 Win32 纪元 (1601-01-01),示例时间戳是 13224382394716547600ns(或者更确切地说,是 132243823947165476 100ns 单位)。这是我尝试过的:测试 1(添加纳秒)win_epoch := time.Date(1601,1,1, 0, 0, 0, 0, time.UTC)current_ts_1 := win_epoch.Add(13224382394716547600*time.Nanosecond)测试 2(添加天数)win_epoch := time.Date(1601,1,1, 0, 0, 0, 0, time.UTC)current_ts_2 := win_epoch.AddDate(0,0,(132243823947165476/10000000 / 3600 / 24))current_ts_1在测试 1 中失败并出现溢出错误。current_ts_2在测试 2 中,我只提供了日期级别的解决方案。理想情况下,我能够获得毫秒级的分辨率。是否存在不溢出传递给的持续时间.Add()并仍然获得此分辨率的方法?
查看完整描述

1 回答

?
三国纷争

TA贡献1804条经验 获得超7个赞

这种转换需要处理一些巨大的数字。您的示例时间 13,224,382,394,716,547,600 太大而无法放入 int64(最大 9,223,372,036,854,775,807),但确实适合 unit64。


处理此问题时不能使用 time.Duration ,因为这是一个 int64 纳秒计数;正如文档所说,“该表示将最大可表示的持续时间限制为大约 290 年”。这就是您第一次尝试失败的原因。


但是,还有另一种方法可以在这里更好地创建时间func Unix(sec int64, nsec int64) Time。这需要 unix 格式的数据,这是自 1970 年 1 月 1 日 UTC 以来的时间(或 11644473600000000000 自windows epoch以来表示为 ns )。


使用此信息可以执行转换:


func main() {

    const unixTimeBaseAsWin = 11644473600000000000 // The unix base time (January 1, 1970 UTC) as ns since Win32 epoch (1601-01-01) 

    const nsToSecFactor = 1000000000


    timeToConvert := uint64(13224382394716547600)


    unixsec := int64(timeToConvert  - unixTimeBaseAsWin ) / nsToSecFactor

    unixns := int64(timeToConvert  % nsToSecFactor)


    time := time.Unix(unixsec, unixns)

    fmt.Println(time.Local())

}

注意:我已经用一些数字检查了这一点,但建议您在依赖它之前进行进一步测试。


查看完整回答
反对 回复 2022-05-10
  • 1 回答
  • 0 关注
  • 99 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信