golang在defer修改返回值时需要格外注意!
如果未提前声明返回值的话,在defer中修改返回值是无效的操作。原因如下:
package main
func test1() int {
i := 1
defer func() { // 实际则是将局部变量i的地址指针传入,调用runtime.deferproc函数
i++
}()
return i // 将i的值拷贝到调用栈的返回值上
}
func test2() int {
i := 1
defer func(i int) { // 将i的值拷贝一份传入,调用runtime.deferproc函数
i++
}(i)
return i
}
func test3() (i int) { // 提前声明了返回值i
i = 1
defer func() { // 将返回值i的地址指针传入,调用runtime.deferproc函数
i++
}()
return i // 直接将值赋给调用栈的返回值上
}
func main() {
fmt.Println(test1())
fmt.Println(test2())
fmt.Println(test3())
}
这三个方法的返回值
lichenyi@lichenyideMacBook-Pro test % go run test2.go
1
1
2
在test3中函数的返回值提前声明,这使得defer可以直接访问该返回值(因为传入了返回值i的地址指针)
在test1和test2中 defer无法直接访问到返回值(因为这两种情况传入的分别是局部变量i的指针和拷贝值,defer虽然正确运行了,但是test2和test3的函数栈在函数返回后就销毁了,所以局部变量i也销毁了,因此对于调用函数而言是无感知的,所以没有成功修改返回值)
————————————————
版权声明:本文为CSDN博主「李晨毅」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/weixin_40486544/article/details/108741704