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 可以修改命名返回值,不能修改匿名返回值
相关推荐
2302_809798327 分钟前
【JavaWeb】Docker项目部署
java·运维·后端·青少年编程·docker·容器
蓝婷儿12 分钟前
6个月Python学习计划 Day 16 - 面向对象编程(OOP)基础
开发语言·python·学习
孔令飞24 分钟前
Kubernetes 节点自动伸缩(Cluster Autoscaler)原理与实践
ai·云原生·容器·golang·kubernetes
渣渣盟28 分钟前
基于Scala实现Flink的三种基本时间窗口操作
开发语言·flink·scala
zhojiew30 分钟前
关于akka官方quickstart示例程序(scala)的记录
后端·scala
sclibingqing42 分钟前
SpringBoot项目接口集中测试方法及实现
java·spring boot·后端
糯米导航1 小时前
Java毕业设计:办公自动化系统的设计与实现
java·开发语言·课程设计
糯米导航1 小时前
Java毕业设计:WML信息查询与后端信息发布系统开发
java·开发语言·课程设计
MessiGo1 小时前
Javascript 编程基础(5)面向对象 | 5.1、构造函数实例化对象
开发语言·javascript·原型模式
大霞上仙1 小时前
nonlocal 与global关键字
开发语言·python