今天来水一篇,最近比较忙,一直没有时间写 go 相关的,今天从一个小问题入手,来说说 struct 的比较问题。
由于已经有很多其他的文章说过这个问题,我这里赘述就显得多余,所以我直接给出结论,并直接说明在实际中用的上的。
原本这应该是某人想出的面试题,但是如果光光是解决这个问题的话,太应试了。大白话就是,谁没事去比较两个 struct 呢?为什么要比较呢?
那比较的原因,肯定是我们需要知道两个结构体是否相等。
两个结构体是否相等,比价的依据有两个:
重点1:如果两个 struct 类型不同,一定是无法比较的,会直接编译报错,也没有人这么干吧。。
其实大多数情况下,我们不需要知道这个问题的答案,至少我无法想象到为什么要比较两个对象的地址是不是一样。
package main
import "fmt"
type User struct {
Name string
}
func main() {
u := &User{Name: "star"}
u2 := &User{Name: "star"}
u3 := u
fmt.Println(u == u2) // false
fmt.Println(u == u3) // true
}
当然比较结果是符合我们正常心里预期的,两个变量都指向同一个地址的 struct,当然相等,其实这本质也就是指针的比较,只要指向相同变量就相等了。很简单。
这个是实际中确实会使用到的情况,我们有可能需要比较两个结构体中的内容是否完全一致,那么我们是否也可以使用 ==
来进行比较呢?
package main
import "fmt"
type User struct {
Name string
}
func main() {
u := User{Name: "star"}
u2 := User{Name: "star"}
u3 := u
fmt.Println(u == u2) // true
fmt.Println(u == u3) // true
}
输出很正常,好像可以?但,其实不然
重点 2:当结构体内存在不可比较的类型时(slice、map、function),使用
==
比较会报错。
那么当出现这种情况时我们需要使用 reflect.DeepEqual
方法进行比较:
package main
import (
"fmt"
"reflect"
)
type User struct {
Name string
Books []string
}
func main() {
u := User{Name: "star", Books: []string{"a", "b"}}
u2 := User{Name: "star", Books: []string{"a", "b"}}
// fmt.Println(u == u2) // Invalid operation: u == u2 (the operator == is not defined on User)
fmt.Println(reflect.DeepEqual(u, u2)) // true
}
有了它,那么两个 struct 就可以比较内容了,并且满足我们的要求。
其实结论很简单,当我们需要比较两个 struct 内容时,最好使用 reflect.DeepEqual
方法进行比较,这样无论什么类型均可满足我们的比较要求。
最后来按官方的话回答一下标题的问题: