关键字 defer ⽤于延迟一个函数或者方法(或者当前所创建的匿名函数)的执行。
注意,defer语句只能出现在函数或方法的内部。
defer语句经常被用于处理成对的操作,如打开、关闭、连接、断开连接、加锁、释放锁。通过defer机制,不论函数逻辑多复杂,都能保证在任何执行路径下,资源被释放。
释放资源的defer应该直接跟在请求资源的语句后。
func main() {
fmt.Println("this is a test")
defer fmt.Println("this is a defer") //main结束前调用
/*
运行结果:
this is a test
this is a defer
*/
}
多个defer
会按栈的顺序,后进先出。
func test(x int) {
fmt.Println(100 / x)//x为0时,产生异常
}
func main() {
defer fmt.Println("aaaaaaaa")
defer fmt.Println("bbbbbbbb")
defer test(0)
defer fmt.Println("cccccccc")
/*
运行结果:
cccccccc
bbbbbbbb
aaaaaaaa
panic: runtime error: integer divide by zero
*/
}
panic也挡不出defer的执行。所以可以拿来做兜底。
但是要注意把兜底的defer放在最上面。
package main
import "fmt"
func test(x int) {
defer fmt.Println("我要崩溃了")
fmt.Println(100 / x) //x为0时,产生异常
}
func main() {
defer func() {
recover()
fmt.Println("没事,我兜住了")
}()
defer test(0)
defer fmt.Println("aaaaaaaa")
defer fmt.Println("bbbbbbbb")
/*
运行结果:
bbbbbbbb
aaaaaaaa
我要崩溃了
没事,我兜住了
*/
}
如果panic在上面呢?
package main
import "fmt"
func main() {
panic(11) //这里就直接崩溃退出了,兜不住的。
defer func() {
recover()
fmt.Println("没事,我兜住了")
}()
}