golang中defer的关键特性示例详解

作者:三月沙 时间:2023-08-06 06:12:45 

前言

大家都知道golang的defer关键字,它可以在函数返回前执行一些操作,最常用的就是打开一个资源(例如一个文件、数据库连接等)时就用defer延迟关闭改资源,以免引起内存泄漏。本文主要给大家介绍了关于golang中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

来源:http://sanyuesha.com/2017/07/23/go-defer/

标签:golang,defer,关键特性
0
投稿

猜你喜欢

  • 服务器XMLHTTP(Server XMLHTTP in ASP)基础

    2008-11-11 12:45:00
  • 解决PyCharm 中写 Turtle代码没提示以及标黄的问题

    2023-05-12 09:08:30
  • Python关键字之global与nonlocal

    2023-12-09 13:34:18
  • python爬虫headers设置后无效的解决方法

    2021-09-04 00:55:17
  • asp javascript值的互相传递方法

    2011-03-30 10:37:00
  • CSS的书写顺序规范

    2008-06-12 13:51:00
  • innodb_flush_method取值方法(实例讲解)

    2024-01-13 10:12:44
  • JSP自定义标签Taglib实现过程重点总结

    2024-03-15 23:49:46
  • 解决pytorch中的kl divergence计算问题

    2023-11-12 11:02:00
  • Apache SkyWalking 监控 MySQL Server 实战解析

    2024-01-24 01:31:54
  • python实现简易的学生信息管理系统

    2021-08-24 12:07:09
  • python 读写excel文件操作示例【附源码下载】

    2023-02-20 09:26:31
  • keras 自定义loss损失函数,sample在loss上的加权和metric详解

    2021-06-03 08:11:33
  • Python中schedule模块定时任务的使用方法(2)

    2023-09-01 09:59:59
  • python中如何写类

    2023-09-02 18:19:58
  • 用XML和XSL来生成动态页面

    2008-09-04 10:35:00
  • mysql 8.0.27 安装配置方法图文教程(Windows64位)

    2024-01-19 07:37:59
  • tensorflow模型继续训练 fineturn实例

    2023-07-10 12:53:09
  • python工具——Mimesis的简单使用教程

    2021-08-06 07:27:23
  • Python利用Canny算法检测硬币边缘

    2022-05-15 08:04:43
  • asp之家 网络编程 m.aspxhome.com