关键字 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("没事,我兜住了")
	}()

}

【Go语言学习笔记】Go的defer_释放资源