struct能不能比较? 很显然这句话包含了两种情况:
type S struct {
Name string
Age int
Address *int
}
func main() {
a := S{
Name: "aa",
Age: 1,
Address: new(int),
}
b := S{
Name: "aa",
Age: 1,
Address: new(int),
}
fmt.Println(a == b)
}
运行上面的代码发现会打印false。既然能正常打印输出,说明是可以个比较的,接下来让我们来个死亡两问
什么可以比较?
回到上面的划重点部分,在总结中我们可以知道,golang中 Slice,Map,Function 这三种数据类型是不可以直接比较的。我们再看看S结构体,该结构体并没有包含不可比较的成员变量,所以该结构体是可以直接比较的。
为什么打印输出false?
a 和 b 虽然是同一个struct 的两个实例,但是因为其中的指针变量 Address 的值不同,所以 a != b,如果a b 在初始化时把 Address 去掉(不给 Address 初始化),那么这时 a == b 为true, 因为ptr变量默认值是nil,又或者给 Address 成员变量赋上同一个指针变量的值,也是成立的。
如果给结构体S增加一个Slice类型的成员变量后又是什么情况呢?
type S struct {
Name string
Age int
Address *int
Data []int
}
func main() {
a := S{
Name: "aa",
Age: 1,
Address: new(int),
Data: []int{1, 2, 3},
}
b := S{
Name: "aa",
Age: 1,
Address: new(int),
Data: []int{1, 2, 3},
}
fmt.Println(a == b)
}
这时候会打印输出什么呢?true?false?实际上运行上面的代码会报下面的错误:
./test.go:37:16: invalid operation: a == b (struct containing []int cannot be compared)
a, b 虽然是同一个struct两个赋值相同的实例,因为结构体成员变量中带有了不能比较的成员(slice),是不可以直接用 == 比较的,所以只要写 == 就报错
同一个struct的两个实例可比较也不可比较,当结构不包含不可直接比较成员变量时可直接比较,否则不可直接比较
但在平时的实践过程中,当我们需要对含有不可直接比较的数据类型的结构体实例进行比较时,是不是就没法比较了呢?事实上并非如此,golang还是友好滴,我们可以借助 reflect.DeepEqual 函数 来对两个变量进行比较。所以上面代码我们可以这样写:
type S struct {
Name string
Age int
Address *int
Data []int
}
func main() {
a := S{
Name: "aa",
Age: 1,
Address: new(int),
Data: []int{1, 2, 3},
}
b := S{
Name: "aa",
Age: 1,
Address: new(int),
Data: []int{1, 2, 3},
}
fmt.Println(reflect.DeepEqual(a, b))
}
结论:可以比较,也不可以比较
可通过强制转换来比较:
type T2 struct {
Name string
Age int
Arr [2]bool
ptr *int
}
type T3 struct {
Name string
Age int
Arr [2]bool
ptr *int
}
func main() {
var ss1 T2
var ss2 T3
// Cannot use ‘ss2‘ (type T3) as type T2 in assignment
//ss1 = ss2 // 不同结构体之间是不可以赋值的
ss3 := T2(ss2)
fmt.Println(ss3==ss1) // true
}
如果成员变量中含有不可比较成员变量,即使可以强制转换,也不可以比较
struct必须是可比较的,才能作为key,否则编译时报错
type person struct {
Name string
Age int
Address []string
}
func main() {
p1 := person{Name: "wanghao", Age: 30}
m := make(map[person]bool)
m[p1] = true
fmt.Println(m)
}
编译时会报错 去掉Address就可
# command-line-arguments
usercode/file.go:15: invalid map key type person
原文:https://www.cnblogs.com/wanghaostec/p/15085486.html