一、defer关键字的特性:
(1)延迟执行:defer关键字后的函数都是在整个函数执行结束return之后才执行的。
(2)参数预计算:defer函数的形参会在定义时就完成了该参数的拷贝。
(3)FILO:先进后出,若多个defer函数在同一函数内,执行顺序遵循先进后出原理。即第一个defer函数最后一个被执行
二、defer中的坑
func main() {
fmt.Println(f1())
fmt.Println(f2())
fmt.Println(f3())
fmt.Println(f4())
}
// return 10
// 因为返回值 r 已经被声明,return不会生成一个临时变量,直接在 变量r上进行+5,最终返回10
func f1() (r int) {
r = 5
defer func() {
r = r + 5
}()
return r
}
// return 5
// defer后面的函数是一个匿名函数,在匿名函数中,参数r是以值的方式,也就是r的副本,所以最终返回的r不会增加5
// 假设defer中函数的参数为指针类型,比如 (r *int),则会改变最终结果
func f2() (r int) {
r = 5
defer func(r int) {
r = r + 5
}(r)
return r
}
// f3和f4都是 return5
// return r并不是原子操作,可以分为两个步骤,第一步先拷贝一个临时变量,将r的值拷贝到这个临时变量,第二步才是返回这个变量的值
// defer函数执行时机在第一步和第二步之间,所以后续defer的修改r,不会对最终这个临时变量的值产生影响
func f3() int {
r := 5
defer func() {
r = r + 5
}()
return r
}
func f4() int {
r := 5
defer func(r int) {
r = r + 5
}(r)
return r
}