golang defer 详解

简介: golang defer 详解

本文介绍了Go语言中defer的关键作用,包括资源清理(如文件关闭),执行recover以捕获panic,以及函数参数在defer时的确定性。还探讨了多个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 的函数参数已经进行执行计算并确定了。

再看一个例子:

package main

import (
  "fmt"
)

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 的函数可以读取和修改带名称的返回值

package main

import "fmt"

func main() {
  fmt.Println(c())
}

func c() (i int) {
  defer func() { i++ }()
  return 1
}

defer 的函数是在 return 之后执行的,可以修改带名称的返回值,上面的函数 c 返回的是 2


目录
相关文章
|
1月前
|
Go Python
通过 atexit 模块让 Python 实现 Golang 的 defer 功能
通过 atexit 模块让 Python 实现 Golang 的 defer 功能
18 2
|
3月前
|
Serverless Go
Golang 开发函数计算问题之defer 中的 recover() 没有捕获到 如何解决
Golang 开发函数计算问题之defer 中的 recover() 没有捕获到 如何解决
|
6月前
|
Go 开发者
Golang深入浅出之-Go语言 defer、panic、recover:异常处理机制
Go语言中的`defer`、`panic`和`recover`提供了一套独特的异常处理方式。`defer`用于延迟函数调用,在返回前执行,常用于资源释放。它遵循后进先出原则。`panic`触发运行时错误,中断函数执行,直到遇到`recover`或程序结束。`recover`在`defer`中捕获`panic`,恢复程序执行。注意避免滥用`defer`影响性能,不应对可处理错误随意使用`panic`,且`recover`不能跨goroutine捕获panic。理解并恰当使用这些机制能提高代码健壮性和稳定性。
125 2
|
安全 Go
Golang 语言中的 defer 怎么使用?
Golang 语言中的 defer 怎么使用?
45 0
|
6月前
|
Java Go
浅谈Golang 不同版本的defer
浅谈Golang 不同版本的defer
55 1
|
6月前
|
存储 编译器 测试技术
Golang底层原理剖析之defer
Golang底层原理剖析之defer
70 0
|
存储 Java 编译器
Golang中的defer(2)
Golang中的defer(2)
4717 2
|
编译器 Go
Golang中的defer
Golang中的defer
65 0
|
Go
Golang中的defer(1)
Golang中的defer(1)
77 0