一、defer 的作用和执行时机
go 的 defer 语句是用来延迟执行函数的,而且延迟发生在调用函数 return之后,比如
func a() int { defer b() return 0 }
b 的执行是发生在return 0之后,注意defer
的语法,关键字defer
之后是函数的调用。
二、defer 的重要用途一:清理释放资源
由于defer
的延迟特性,defer
常用在函数调用结束之后清理相关的资源,比如
f, _ := os.Open(filename) defer f.Close()
文件资源的释放会在函数调用结束之后借助defer
自动执行,不需要时刻记住哪里的资源需要释放,打开和释放必须相对应。
用一个例子深刻诠释一下defer
带来的便利和简洁。
代码的主要目的是打开一个文件,然后复制内容到另一个新的文件中,没有defer
时这样写:
func CopyFile(dstName, srcName string) (written int64, err error) { src, err := os.Open(srcName) if err != nil { return } dst, err := os.Create(dstName) if err != nil { //1 return } written, err = io.Copy(dst, src) dst.Close() src.Close() return }
代码在#1
处返回之后,src
文件没有执行关闭操作,可能会导致资源不能正确释放,改用defer
实现:
func CopyFile(dstName, srcName string) (written int64, err error) { src, err := os.Open(srcName) if err != nil { return } defer src.Close() dst, err := os.Create(dstName) if err != nil { return } defer dst.Close() return io.Copy(dst, src) }
src
和dst
都能及时清理和释放,无论return
在什么地方执行。
鉴于defer
的这种作用,defer
常用来释放数据库连接,文件打开句柄等释放资源的操作。
###defer
的重要用途二:执行recover
被defer
的函数在return
之后执行,这个时机点正好可以捕获函数抛出的panic
,因而defer
的另一个重要用途就是执行recover
。
recover
只有在defer
中使用才更有意义,如果在其他地方使用,由于program
已经调用结束而提前返回而无法有效捕捉错误。
package main import ( "fmt" ) func main() { defer func() { if ok := recover(); ok != nil { fmt.Println("recover") } }() panic("error") }
记住defer
要放在panic
执行之前。
三、多个 defer 的执行顺序
defer 的作用就是把关键字之后的函数执行压入一个栈中延迟执行,多个defer
的执行顺序是后进先出LIFO
:
defer func() { fmt.Println("1") }() defer func() { fmt.Println("2") }() defer func() { fmt.Println("3") }()
输出顺序是 321。
这个特性可以对一个array
实现逆序操作。
四、被 deferred 函数的参数在 defer 时确定
这是defer
的特点,一个函数被defer
时,它的参数在defer
时进行计算确定,即使defer
之后参数发生修改,对已经defer
的函数没有影响,什么意思?看例子:
func a() { i := 0 defer fmt.Println(i) i++ return }
a 执行输出的是 0 而不是 1,因为defer
时,i 的值是 0,此时被defer
的函数参数已经进行执行计算并确定了。
再看一个例子:
func calc(index string, a, b int) int { ret := a + b fmt.Println(index, a, b, ret) return ret } func main() { a := 1 b := 2 defer calc("1", a, calc("10", a, b)) a = 0 return }
执行代码输出
10 1 2 3 1 1 3 4
defer
函数的参数 第三个参数在defer
时就已经计算完成并确定,第二个参数 a 也是如此,无论之后 a 变量是否修改都不影响。
五、被defer的函数可以读取和修改带名称的返回值
func c() (i int) { defer func() { i++ }() return 1 }
被defer
的函数是在return
之后执行,可以修改带名称的返回值,上面的函数 c 返回的是 2。
六、defer、return的执行顺序
https://www.cnblogs.com/l199616j/p/15500631.html
参考:https://zhuanlan.zhihu.com/p/60005467