我不知道 OPCUA 协议,但您的六进制时间戳可以转换为 Go 时间戳,如下所示:
首先,必须向后读取时间戳的字节(与您使用的字节顺序不同)。
所以代替这些:
0x1c67dc4ab30dd201, 0x15605a199070d201, 0x322b4f629970d201
您必须反转字节并使用这些:
0x01d20db34adc671c, 0x01d27090195a6015, 0x01d27099624f2b32
接下来,这些值以10 微秒为单位,因此要获得纳秒,请乘以 100。
最后,与1970 年 1 月 1 日 UTC 的 Epoch 相比,偏移量为369 年。
最终转换:
var delta = time.Date(1970-369, 1, 1, 0, 0, 0, 0, time.UTC).UnixNano()
func convert(t int64) time.Time {
// reverse bytes:
var t2 int64
for i := 0; i < 8; i, t = i+1, t>>8 {
t2 = t2<<8 | t&0xff
}
return time.Unix(0, t2*100+delta)
}
测试它:
// If you want results in CET:
cet, err := time.LoadLocation("CET")
if err != nil {
panic(err)
}
ts := []int64{0x1c67dc4ab30dd201, 0x15605a199070d201, 0x322b4f629970d201}
for _, t := range ts {
fmt.Println(convert(t).In(cet))
}
输出(在Go Playground上试试):
2016-09-13 13:38:05.3431068 +0200 CEST
2017-01-17 08:05:35.0120469 +0100 CET
2017-01-17 09:12:02.8828466 +0100 CET
处理[]byte
输入
[]byte
如果时间戳作为字节切片io.Reader
(您可以使用以下转换器功能:
func convert2(data []byte) time.Time {
t := binary.LittleEndian.Uint64(data)
return time.Unix(0, int64(t)*100+delta)
}
测试它:
data := []byte{0x32, 0x2b, 0x4f, 0x62, 0x99, 0x70, 0xd2, 0x01}
fmt.Println(convert2(data).In(cet))
输出(这个也在上面链接的Playground 示例中):
2017-01-17 09:12:02.8828466 +0100 CET