你在学习Golang相关的知识吗?本文《go map value 和 nil 的区别》,主要介绍的内容就涉及到,如果你想提升自己的开发能力,就不要错过这篇文章,大家要知道编程理论基础和实战操作都是不可或缺的哦!
问题内容在go中,当key不存在时,map的值是零值。 我下面有一个简短的代码片段: 演示
package main
import (
"sync"
)
func main() {
var mm map[int]sync.Mutex
var m sync.Mutex
mm[1].Lock() // not work due to cannot call pointer method on mm[1] and cannot take the address of mm[1]
m.Lock() // work normal
}
上面的mm[1]
和m
有什么区别?我用 reflect 来检查,但看不出它们之间的区别。关于造成差异的原因有任何线索吗?
解决方案
问题不在于映射的零值,而在于方法调用期间的可寻址性。
互斥锁上的 Lock 方法有一个指针接收器:
func (*mutex) lock
给定变量 msync.mutex
,使用指针接收器调用它的方法将自动变成 &m.lock()
,根据 spec:
如果 x(的类型)的方法集,则方法调用 x.m() 有效 包含 m 并且参数列表可以赋值给形参列表 米。如果 x 是可寻址的并且 &x 的方法集包含 m,则 x.m() 是 (&x).m() 的简写
其中重要的部分是 addressable 要求。变量 m
是可寻址的,但映射查找的返回值不是可寻址的。这意味着编译器不会尝试调用 &mm[1].lock()
。
这些可以在您尝试编译示例时返回的错误消息中看到:
./prog.go:11:7: cannot call pointer method on mm[1]
./prog.go:11:7: cannot take the address of mm[1]
今天关于《go map value 和 nil 的区别》的内容介绍就到此结束,如果有什么疑问或者建议,可以在编程网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!