goland defer
朋友询问golang中defer的一些问题,作下记录
defer与return的顺序
首先是确定defer
和return
谁现执行。在网上或者官网上去查询可以找到答案。
defer
是在return
之前执行。
多个defer的顺序
defer在golang中的调用顺序类似于栈,先进后出
例子
- 函数返回值为已命名和任意
func exa3()(res int){
defer func(){
res = res + 3
}()
return 1
}
func exa4() int {
res :=1
defer func(){
res = res + 3
}()
return res
}
分别执行上述代码,结果为 4和1。
在官方文档上可以查询If the deferred function has any return values, they are discarded when the function completes
也就是说任意返回defer函数执行完毕的东西会被丢弃。
- 作用域 1
func exa3()(res int){
res = 1
defer func(res int){
res = res + 3
}(res)
return 1
}
func exa4()(res int){
res = 1
defer func(){
res = res + 3
}()
return 1
}
再次执行上述代码,这次2个函数的返回值都是已命名参数。结果为1,4。
在exa3中 执行defer时传入了int类型的参数,golang中int类型的传递是一次拷贝,所以defer改变的参数已不再是原本参数。
在exa4中 执行defer函数时,利用了闭包原理,defer函数读取了exa4函数内部变量,并修改了exa4函数变量的值
- 作用域 2
func exa5() (s [4]int) {
s = [4]int{1, 2, 3, 4}
defer func(s [4]int) {
s[0] = 4
}(s)
defer func() {
s[0] = 3
}()
s[0] = 2
return s
}
func exa6() (s []int) {
s = []int{1, 2, 3, 4}
defer func(s []int) {
s[0] = 4
}(s)
defer func() {
s[0] = 3
}()
s[0] = 2
return s
}
再次执行上述代码 结果分别为[3,2,3,4]和[4,2,3,4]
在exa5中 golang对数组的传递也是一次拷贝。
在exa6中 golang对切片的传递是指针的传递,而第1个defer晚于第2个defer执行。
- 作用域
func exa8()(res *int){
var i = 1
res = &i
defer func(res *int){
*res = *res + 3
}(res)
return
}
那么执行上述代码的结果是多少呢。