深入理解 Go-Defer的机制

简介: defer 的作用和执行时机go 的 defer 语句是用来延迟执行函数的,而且延迟发生在调用函数 return 之后,比如func a() int { defer b() return 0}b 的执行是发生在return 0之后,注意defer 的语法,关键字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。

参考资料
https://blog.golang.org/defer-panic-and-recover

相关文章
|
7月前
|
Go 调度
GO语言函数的内部运行机制分析
以上就是Go语言中函数的内部运行机制的概述,展示了函数在Go语言编程中如何发挥作用,以及Go如何使用简洁高效的设计,使得代码更简单,更有逻辑性,更易于理解和维护。尽管这些内容深入了一些底层的概念,但我希望通过这种方式,将这些理论知识更生动、更形象地带给你,让你在理解的同时找到编程的乐趣。
132 5
|
7月前
|
安全 Go
defer关键字:延迟调用机制-《Go语言实战指南》
Go 语言中的 `defer` 是用于延迟执行函数调用的关键字,广泛应用于资源释放、异常捕获和日志记录等场景。它在函数返回前执行,支持栈式后进先出(LIFO)顺序,参数求值时机为声明时而非执行时。常见用法包括文件关闭、锁解锁及结合 `recover` 处理 panic。尽管高效,频繁使用可能带来性能开销,需谨慎处理。总结而言,`defer` 是构建健壮代码的核心工具之一。
|
8月前
|
存储 算法 物联网
解析局域网内控制电脑机制:基于 Go 语言链表算法的隐秘通信技术探究
数字化办公与物联网蓬勃发展的时代背景下,局域网内计算机控制已成为提升工作效率、达成设备协同管理的重要途径。无论是企业远程办公时的设备统一调度,还是智能家居系统中多设备间的联动控制,高效的数据传输与管理机制均构成实现局域网内计算机控制功能的核心要素。本文将深入探究 Go 语言中的链表数据结构,剖析其在局域网内计算机控制过程中,如何达成数据的有序存储与高效传输,并通过完整的 Go 语言代码示例展示其应用流程。
143 0
|
存储 安全 Go
Go 语言以其高效的并发编程能力著称,主要依赖于 goroutines 和 channels 两大核心机制
Go 语言以其高效的并发编程能力著称,主要依赖于 goroutines 和 channels 两大核心机制。本文介绍了这两者的概念、用法及如何结合使用,实现任务的高效并发执行与数据的安全传递,强调了并发编程中的注意事项,旨在帮助开发者更好地掌握 Go 语言的并发编程技巧。
176 2
|
存储 编译器 Go
掌握 Go 语言的 defer 关键字
掌握 Go 语言的 defer 关键字
|
Go 数据处理
实验深度理解Go中try...catch...的panic、defer、recover用法
文章通过实验代码演示了Go语言中如何使用panic、defer和recover函数来模拟try...catch...的异常处理机制,并详细解释了每个函数的作用和在异常处理中的使用场景。
178 0
|
Java Go
Go 中 slice 的内存管理机制
Go 中 slice 的内存管理机制
205 2