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:

vbnet 复制代码
package main

import "fmt"

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

输出结果:

Go 复制代码
Loop ended
2
1
0

栗子2:

Go 复制代码
package main

import "fmt"

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

输出结果:

Go 复制代码
Loop ended
3
3
3

造成这种结果的原因:

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

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

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

4.defer与返回值的关系

栗子3:

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

栗子4:

Go 复制代码
func bar() (i int) {    // 返回值命名了
    i = 0
    defer 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 可以修改命名返回值,不能修改匿名返回值
相关推荐
辞旧 lekkk4 小时前
【Qt】信号和槽
linux·开发语言·数据库·qt·学习·mysql·萌新
2zcode5 小时前
运动模糊图像复原的MATLAB仿真与优化
开发语言·matlab
无风听海5 小时前
深入剖析 YARP 的 Transforms:构建灵活的反向代理转换管道
后端·中间件·asp.net
袁雅倩19975 小时前
当吸尘器、筋膜枪都用上Type-C,供电方案该怎么选?浅谈PD取电芯片ECP5702的应用
c语言·开发语言·支持向量机·动态规划·推荐算法·最小二乘法·图搜索算法
Gopher_HBo5 小时前
负载均衡
后端
自由生长20246 小时前
RAG已死?什么标题党啊!
后端
Aaswk6 小时前
Java Lambda 表达式与流处理
java·开发语言·python
万邦科技Lafite6 小时前
京东item_get接口实战案例:实时商品价格监控全流程解析
java·开发语言·数据库·python·开放api·淘宝开放平台
东方小月6 小时前
5分钟搞懂Harness Engineering(驾驭工程):从提示词到AI Agent的进化之路
前端·后端·架构
Cyber4K7 小时前
【Python专项】进阶语法-系统资源监控与数据采集(1)
开发语言·python·php