defer会经常用到,但是这个坑对于我并不会踩到,因为我通常不会用到命名返回参数,一则没有多少必要,二则增加了代码阅读难度。不过,这个坑能让人很好地理解return这个关键字,所以在此记录下。
func test() (res int) {
res = 1
defer func() {
res++
}()
return 0
}
默默在心里运行这个程序,第一个返回值应该是0嘛,其实不然,return并不是个原子操作。分为了,赋值和返回两个操作,<return 0>其实是等价于<res = 0, return res>,而defer是在赋值操作后执行的,所以这个函数等价于:
func test() (res int) {
res = 1
res = 0 //赋值
func() {
res++ //defer
}()
return res //返回
}
再来看个例子:
func test() (res int) {
tmp := 1
defer func() {
tmp++
}()
return tmp
}
按刚才的经验分析下这个程序,tmp=1,然后返回前又加了1,应该是2吧。其实又错了,注意返回值命名是res,而不是tmp,defer改变的只能是tmp。这个函数可以等价于:
func test() (res int) {
tmp := 1
res = tmp
func() {
tmp++
}()
return
}
结果应该是1。