当前位置: 首页 > news >正文

go语言中defer使用指南

目录

1.使用场景

2.执行顺序

3.for循环中的defer及defer中的闭包陷阱

4.defer与返回值的关系

5.总结


1.使用场景

在编程的时候,经常需要打开一些资源,比如数据库连接、文件、锁等,这些资源需要在用完之后释放掉,否则会造成内存泄漏。

在 Go 中 defer 一般用于异常处理、资源释放、文件关闭、解锁互斥量等操作。有一个编码好习惯就是,在初始化资源后,可以在后面紧跟一个defer函数取释放资源。

2.执行顺序

defer执行顺序是后进先出(底层是由链栈实现的)。

在函数内,defer与return的执行顺序是:defer在函数内return执行之后、函数退出之前执行

3.for循环中的defer及defer中的闭包陷阱

闭包:一个匿名函数捕获了外部变量,就形成了闭包

Go 语言 中,如果在循环内执行 defer 语句,每次循环中执行的 defer 语句都会将其延迟的函数调用压入栈中。所有的 defer 调用都会在循环所在的函数返回之前按照后进先出(LIFO) 顺序执行。

还有超级重要,即defer 声明时函数的参数会立即求值(这针对的是有参数传入的函数),而不是等到延迟调用时才求值。

重要的事情说三遍,defer 声明时函数的参数会立即求值,deferzai 声明时函数的参数会立即求值,defer 声明时函数的参数会立即求值!!!

什么是函数的参数会立即求值?

栗子1:

package mainimport "fmt"func main() {for i := 0; i < 3; i++ {defer func(val int) {    // 闭包,捕获了外部变量ifmt.Println(val) // 打印传递的值}(i) // 在这里立即捕获当前的 i 值,}fmt.Println("Loop ended")
}

输出结果:

Loop ended
2
1
0

 栗子2: 

package mainimport "fmt"func main() {for i := 0; i < 3; i++ {defer func() {   // 闭包,捕获了外部变量ifmt.Println(i) // 直接引用 i,持有的是引用,而非具体的值}()}fmt.Println("Loop ended")
}

输出结果:

Loop ended
3
3
3

造成这种结果的原因:

闭包捕获的是引用,而非具体的值!!!这一点很重要

栗子1:通过参数传递固定 i 的值,每个匿名函数保存的是当前循环中 i 的值。因此,输出结果是 2, 1, 0

栗子2:匿名函数捕获的是循环变量 i 的引用,所有匿名函数访问的都是同一个变量 i,而此时 i 的值已经是 3。因此,输出结果是 3, 3, 3

 

4.defer与返回值的关系

 栗子3:        

func foo() int {    //返回值未命名i := 0defer func() { i++ }()return i
}    // 返回0

栗子4:

func bar() (i int) {    // 返回值命名了i = 0defer func() { i++ }()return i
}    // 返回1

 造成这种结果的原因是:

defer可以修改命名返回值,不能修改匿名返回值!!!

  • foo函数中,由于使用的是匿名返回值i的值在return i的时候就已经确定为0。即使随后的defer函数将i增加到1,这个改变不会反映在已经确定的返回值上,所以foo函数返回0。

  • bar函数中,由于使用了命名返回值idefer函数能够访问并修改这个命名返回值。因此,在return i之后但函数真正返回之前,defer函数将i从0增加到1,导致bar函数最终返回1。

5.总结

  1. defer  一般用于异常处理、资源释放、文件关闭、解锁互斥量
  2. defer 在函数内return执行之后、函数退出之前执行
  3. defer 声明时函数的参数会立即求值(针对有参数传入的函数)
  4. 闭包捕获的是引用,而非具体的值
  5. defer 可以修改命名返回值,不能修改匿名返回值

相关文章:

  • 逻辑思维:从混沌到秩序的理性推演在软件开发中的应用
  • 使用Nacos 打造微服务配置中心
  • Go语言之sync包 WaitGroup的使用和底层实现
  • 文件操作函数
  • 基于cubeMX的hal库STM32实现硬件IIC通信控制OLED屏
  • 汽车VIN码识别:解锁汽车行业的智能密码
  • Spark-SQL 项目
  • 爬虫(requests库,logging库)
  • react 父子组件通信 子 直接到父, 父 forwardref子
  • window上 elasticsearch v9.0 与 jmeter5.6.3版本 冲突,造成es 启动失败
  • 关于在Springboot中设置时间格式问题
  • Git -> Git 所有提交阶段的回滚操作
  • 测试-时间规模化定律可以改进世界基础模型吗?
  • [Java · 铢积寸累] 数据结构 — 二维数组 - 概念引入
  • 【YOLOv8-pose部署至RK3588】模型训练→转换RKNN→开发板部署
  • docker保存镜像到本地
  • AutoJs相关学习
  • Spring Boot中`logging.config`配置项的详解及使用说明
  • Vscode指定缓存路径 .vscode 路径
  • 嘻游组件解密工具实战教程:资源解包与UI替换全流程
  • “2025上海西九文化周”启动,香港顶尖文艺6月齐聚申城
  • 全品系停货?泸州老窖:暂未接到通知,常规调控手段
  • 外交部:中方在黄海暂定海域建造渔业养殖设施,同中韩海域划界无关
  • 宁德时代与广汽等五车企发布10款巧克力换电新车型:年内将将完成30城1000站计划
  • “未来3天公布”,特朗普俄乌“和平计划”是什么?普京泽连斯基什么态度?
  • 对话地铁读书人|豪宅房产经纪人:读书使我免于抑郁