个人网站设计成品,大连金州代做网站公众号,教育网站建设网,wordpress 后台编辑defer 的作用和执行时机
go 的 defer 是用来延迟执行函数的#xff0c;而且延迟发生在调用函数 return 之后#xff0c;比如#xff1a;
func a() int {defer b()return 0
}b 的执行是发生在 return 0 之后#xff0c;注意 defer 的语法#xff0c;关键字 defer 之后是函…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 { // 1return}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 mainimport (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 : 0defer fmt.Println(i)ireturn
}a 执行输出的是 0 而不是 1因为 defer 时i 的值是 0此时被 defer 的函数参数已经进行执行计算并确定了。
再看一个例子
package mainimport (fmt
)func calc(index string, a, b int) int {ret : a bfmt.Println(index, a, b, ret)return ret
}
func main() {a : 1b : 2defer calc(1, a, calc(10, a, b))a 0return
}执行代码输出:
10 1 2 3
1 1 3 4defer 函数的参数第三个参数在 defer 时就已经计算完成并确定第二个参数 a 也是如此无论之后 a 变量是否修改都不影响。
被 defer 的函数可以读取和修改带名称的返回值
package mainimport fmtfunc main() {fmt.Println(c())
}func c() (i int) {defer func() { i }()return 1
}被 defer 的函数是在 return 之后执行的可以修改带名称的返回值上面的函数 c 返回的是 2。