什么是 defer?如何理解 defer 关键字?Go 中使用 defer 的一些坑。
defer 意为延迟,在 golang 中用于延迟执行一个函数。它可以帮助我们处理容易忽略的问题,如资源释放、连接关闭等。但在实际使用过程中,有一些需要注意的地方(坑),下面我们一一道来。
一些结论
首先,我们来了解 defer 的一些结论:
1、若函数中有多个 defer,其执行顺序为 先进后出,可以理解为栈。
package main
import "fmt"
func main() {
for i := 0; i < 5; i++ {
defer fmt.Println(i)
}
}
Output:
4
3
2
1
0
2、return 会做几件事:
- 给返回值赋值
- 调用 defer 表达式
- 返回给调用函数
package main
import "fmt"
func main() {
fmt.Println(increase(1))
}
func increase(d int) (ret int) {
defer func() {
ret++
}()
return d
}
Output:
2
3、若 defer 表达式有返回值,将会被丢弃。
更多请参考官方文档。
闭包与匿名函数
匿名函数:没有函数名的函数。
闭包:可以使用另外一个函数作用域中的变量的函数。
在实际开发中,defer 的使用经常伴随着闭包与匿名函数的使用。小心踩坑哦:
package main
import "fmt"
func main() {
for i := 0; i < 5; i++ {
defer func() {
fmt.Println(i)
}()
}
}
Output:
5
5
5
5
5
解释一下,defer 表达式中的 i
是对 for 循环中 i
的引用。到最后,i 加到 5,故最后全部打印 5。
如果将 i
作为参数传入 defer 表达式中,在传入最初就会进行求值保存,只是没有执行延迟函数而已。
for i := 0; i < 5; i++ {
defer func(idx int) {
fmt.Println(idx)
}(i) // 传入的 i,会立即被求值保存为 idx
}
巩固一下
为了巩固一下上面的知识点,我们来思考几个例子。
例1:
func f() (result int) {
defer func() {
result++
}()
return 0
}
例2:
func f() (r int) {
t := 5
defer func() {
t = t + 5
}()
return t
}
例3:
func f() (r int) {
defer func(r int) {
r = r + 5
}(r)
return 1
}
例4:
type Test struct {
Max int
}
func (t *Test) Println() {
fmt.Println(t.Max)
}
func deferExec(f func()) {
f()
}
func call() {
var t *Test
defer deferExec(t.Println)
t = new(Test)
}
有没有得出结果?例1的答案不是 0,例2的答案不是 10,例3的答案也不是 6。
例1,比较简单,参考结论2,将 0 赋给 result,defer 延迟函数修改 result,最后返回给调用函数。正确答案是 1。
例2,defer 是在 t 赋值给 r 之后执行的,而 defer 延迟函数只改变了 t 的值,r 不变。正确答案 5。
例3,这里将 r 作为参数传入了 defer 表达式。故 func (r int)
中的 r 非 func f() (r int)
中的 r,只是参数命名相同而已。正确答案 1。
例4,这里将发生 panic。将方法传给 deferExec
,实际上在传的过程中对方法求了值。而此时的 t 任然为 nil
。
参考文档
[1] https://tiancaiamao.gitbooks.io/go-internals/content/zh/03.4.html
[2] http://golang.org/ref/spec#defer_statements
本文链接:https://deepzz.com/post/how-to-use-defer-in-golang.html,参与评论 »
--EOF--
发表于 2017-08-27 02:08:00。
本站使用「署名 4.0 国际」创作共享协议,转载请注明作者及原网址。更多说明 »
提醒:本文最后更新于 2578 天前,文中所描述的信息可能已发生改变,请谨慎使用。
专题「Go 踩坑系列」的其它文章 »
- Go 测试,go test 工具的具体指令 flag (May 20, 2018)
- Go 单元测试,基准测试,http 测试 (May 09, 2018)
- 浅谈 Golang sync 包的相关使用方法 (Aug 19, 2017)
- Golang 博主走过的有关 error 的一些坑 (May 14, 2017)
- Glide命令,如何使用glide,glide.lock (Feb 09, 2017)
- Golang包管理工具Glide,你值得拥有 (Feb 07, 2017)
Comments