1、什么是defer?
defer是Go语言中的延迟执行语句,用来添加函数结束时执行的代码,常用于释放某些已分配的资源、关闭数据库连接、断开socket连接、解锁一个加锁的资源。Go语言机制担保一定会执行defer语句中的代码。其它语言中也有类似的机制,比如Java、C#语言里的finally语句。defer没有嵌套,defer的机制是要取代try except finally
2、defer的使用规则
规则一:当defer被声明时候,其参数就会被实时解析
func a() {
i := 0
defer fmt.Println(i)
i++
return
}
# 输出0
defer后面定义的是一个带变量的函数:fmt.Println(i)。但这个变量(i)在defer被声明的时候,就已经确定其值,换换言之,上面的代码等用于下面的代码
func a() {
i := 0
defer fmt.Println(0)
i++
return
}
# 输出0
规则二:defer执行顺序为先进先出
func b() {
for i := 0; i < 4; i++ {
defer fmt.Print(i)
}
}
#3210
规则三:多个defer语句的执行顺序是逆序执行
当出现多条 defer 语句时以逆序执行(类似栈,即后进先出)。示例代码:
func a() {
i := 0
defer fmt.Println(i)
i++
defer fmt.Println(i)
return
}
# 先输出1,再输出0
规则四:defer可以读取有名返回值
func c() (i int) {
defer func() {
i++
fmt.Println(i)
}()
return 1
}
# 输出2
defer是return调用之后才执行,defer代码块的作用域在函数之内,defer作用域在c函数之内,可以读取i的值
规则五:在panic语句前的defer语句会被执行,在panic语句后面的defer语句不被执行
func panicDefer() {
panic("panic")
defer fmt.Println("defer after panic")
}
# 输出
panic: panic
goroutine 1 [running]:
main.panicDefer()
main.main()
Process finished with exit code 2
可以看到 defer 语句没有执行。
func deferPanic() {
defer fmt.Println("defer before panic")
panic("panic")
}
# 输出
defer before panic
panic: panic
goroutine 1 [running]:
main.deferPanic()
main.main()
Process finished with exit code 2
defer 语句输出了内容。
Go中的panic类似其它语言中的抛出异常,panic后面的代码不再执行(panic语句前面的defer语句会被执行)。
3、defer用法总结
- defer语句在函数返回之前或函数中的return语句之后执行
- 在panic语句后的defer语句不执行
- 多个defer函数的执行顺序是逆序执行:栈规则,先延迟后执行,后延迟先执行
- defer函数传递参数的时候:defer函数调用时,就已经传递了参数数据了,只是暂时不执行函数中的代码而已