深入理解 Go-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 { //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。

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
深入理解Python的TLS机制和Threading.local()
1.背景介绍我之前写过一个关于Python的TLS机制的浅浅析,大家可以参考这个文章,首先,我们再来熟悉熟悉什么是TLS机制。 1.1 Thread Local Storage(线程局部存储)这个概念最早是相对于全局变量来说的,就是我们在编程的时候,会涉及到希望所有线程都能够共享访问同一个变量,在 Python/Go/C 中,我们就可以定义一个全局变量,这样Global Variable 对多个线程就是可见的,因为同一个进程所有线程共享地址空间,大家都可以操作。例如,一个全局的配置变量或单实例对象,所有线程就可以很方便访问了,但是仅仅这样有一个前提,就是这个变量的并发操作必须是幂等的,读写不影响我们程序的正确性。但是往往多线程共同操作一个全局变量,就会影响程序的正确性,因此我们必须枷锁,比如经典的并发加操作。 import threadingcount = 0lock = threading.RLock()def inc(): global count, lock with lock: count += 1 上面那个例子很多博客用来做ThreadLocal变量的讲解,实际上我觉得是有误...
- 下一篇
阿里巴巴开源 Dragonwell JDK 最新版本 8.1.1-GA 发布
导读:新版本主要有三大变化:同步了OpenJDK 上游社区 jdk8u222-ga 的最新更新;带来了正式的 feature:G1ElasticHeap;发布了用户期待的 Windows 实验版本 Experimental Windows version。 距离 Dragonwell JDK 第一个正式版本 8.0.0-GA 发布已经过去 3 个月了,项目在 Github 上的 stars 继续攀升达到了 1900。今天我们带来了最新版本 8.1.1-GA 的发布,包含了全新的特性和更新。详情见下文。 龙井 8.1.1-GA 的新变化 新版本里我们同步了 OpenJDK 上游社区 jdk8u222-ga 的最新更新,带来了上游稳定版本的最新安全更新和补丁。 在 8.0.0-GA 发布的时候,我们介绍了 Dragonwell 第三个新特性
相关文章
文章评论
共有0条评论来说两句吧...