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

与 == 相比,为什么具有相同日期和时间的 2 个时间结构返回 false?

与 == 相比,为什么具有相同日期和时间的 2 个时间结构返回 false?

Go
慕姐4208626 2022-01-04 13:37:53
我有一个time.Time使用time.Date(). 然后我计算1970/1/1 00:00:00.000000000与那个时间之间的纳秒数。然后我用纳秒把它们变成time.Timeusing time.Unix()。但是,如果我将重构时间与原始时间进行比较==,则返回 false。如果我减去这 2 次,则结果持续时间为 0。如果我使用 比较这 2 次time.Equal(),则返回 true。如果我使用time.Date()与第一次相同的值创建另一个时间,则==用于比较这个新时间和原始时间的结果为真。这是演示这一点的代码(Golang Playground):package mainimport (    "fmt"    "time")func main() {    t1 := time.Date(2016, 4, 14, 1, 30, 30, 222000000, time.UTC)    base := time.Date(1970, 1, 1, 0, 0, 0, 0, t1.Location())    nsFrom1970 :=t1.Sub(base).Nanoseconds() // Calculate the number of nanoseconds from 1970/1/1 to t1    t2 := time.Unix(0, nsFrom1970)    fmt.Println(t1)    fmt.Println(t2)    fmt.Println(t1.Sub(t2)) // 0    fmt.Println(t1 == t2) //false    fmt.Println(t1.Equal(t2)) //true    t3 := time.Date(2100, 2, 1, 21, 21, 21, 222000000, time.UTC)    fmt.Println(t1 == t3) //true}为什么重构时间与原始时间相比返回 false?
查看完整描述

1 回答

?
慕容森

TA贡献1853条经验 获得超18个赞

time.Time是一个struct。当您尝试将它们与 进行比较时==,引用Spec:Comparison operator:


如果所有字段都具有可比性,则结构值具有可比性。如果它们对应的非空白字段相等,则两个结构值相等。


因此t1 == t2将比较 2 个Time结构值的所有字段。该Time结构不仅包含自基准时间以来的秒和纳秒,它还包含作为指针的位置:*Location,因此==还将比较位置字段。比较指针:


指针值是可比较的。如果两个指针值指向同一个变量或两者都有 value ,则它们相等nil。指向不同的零大小变量的指针可能相等,也可能不相等。


这就是为什么比较时间与==给出false结果的原因:即使地址不同,2 个位置也可能表示相同的位置,这就是您的情况。


为了证明这一点:


fmt.Println("Locations:", t1.Location(), t2.Location())

fmt.Printf("Location pointers: %p %p\n", t1.Location(), t2.Location())

fmt.Println("Locations equal:", t1.Location() == t2.Location())

输出:


Locations: UTC UTC

Location pointers: 0x1e2100 0x1e6de0

Locations equal: false

这记录在time.Time:


请注意,Go == 运算符不仅比较时间,还比较位置。因此,在没有首先保证为所有值设置了相同的 Location 之前,不应将 Time 值用作地图或数据库键,这可以通过使用 UTC 或 Local 方法来实现。


如果t1andt2也包含相同的*Location指针,即使与==运算符进行比较,它们也将相等。这可以通过对它们调用Time.UTC()或Time.Local()方法来确保,该方法返回使用time.Time相同位置指针 ( *Location) 的值。或者通过使用Time.In()将设置指定位置指针的方法(在正确转换后),例如:


t2 = t2.In(t1.Location())

fmt.Println("Locations:", t1.Location(), t2.Location())

fmt.Printf("Location pointers: %p %p\n", t1.Location(), t2.Location())

fmt.Println("Locations equal:", t1.Location() == t2.Location())

fmt.Println(t1 == t2)     // Now true

fmt.Println(t1.Equal(t2)) // Still true

输出:


Locations: UTC UTC

Location pointers: 0x1e2100 0x1e2100

Locations equal: true

true

true

在Go Playground上试试。


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

添加回答

举报

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