目录
特性1:多个 defer 时的调用顺序:先进后出
特性2:作用域为当前函数,不同函数下拥有不同的 defer 栈
特性4:return 与 defer 执行顺序:return 先 defer 后
特性5:发生 panic 时,已声明的 defer 会出栈执行
首页 后端开发 Golang 用八个demo搞懂Go语言defer的五大特性

用八个demo搞懂Go语言defer的五大特性

Apr 23, 2023 pm 05:40 PM
go语言

在 Go 语言中使用 defer 关键字可以将代码延迟到函数结束之前执行。在开发中,我们经常使用defer关键字完成善后工作,如关闭打开的文件描述符、关闭连接以及释放资源等。

func demo0() {
    fileName := "./test.txt"
    f, _ := os.OpenFile(fileName, os.O_RDONLY, 0)
    defer f.Close()

    contents, _ := ioutil.ReadAll(f)
    fmt.Println(string(contents))}
登录后复制

defer关键字一般紧跟在打开资源代码的后面,防止后续忘记释放资源,defer 声明的代码实际上要等到函数结束之前才会被执行。defer 虽然简单易用,但如果忽略了它的特性,就会在开发中面临困惑。于是,我总结了 defer 的五大特性,通过 8 个demo逐步介绍 defer 的特性。

特性1:多个 defer 时的调用顺序:先进后出

使用多个 defer 关键字时,先被声明的 defer 语句后被调用。类似于“栈”先进后出的特性,defer 的这一特性也很好理解,先被打开的资源,可能会被后续代码依赖,所以要后释放才安全。

func demo1() {
    for i := 0; i < 5; i++ {
        defer fmt.Println("defer:", i)
    }}// defer: 4// defer: 3// defer: 2// defer: 1// defer: 0
登录后复制

特性2:作用域为当前函数,不同函数下拥有不同的 defer 栈

运行 demo2 ,从结果中可以看出,第一个匿名函数和第二个匿名函数的 defer 执行顺序没有关系。
defer 作用域仅为当前函数,在当前函数最后执行,所以不同函数下拥有不同的 defer 栈。

func demo2() {
    func() {
        defer fmt.Println(1)
        defer fmt.Println(2)
    }()

    fmt.Println("=== 新生代农民工啊 ===")

    func() {
        defer fmt.Println("a")
        defer fmt.Println("b")
    }()}// 2// 1// === 新生代农民工啊 ===// b// a
登录后复制

特性3:defer 后的函数形参在声明时确认(预计算参数)

运行 demo3_1 ,根据结果,我们可以得出:defer 在声明时,就已经确认了形参n的值,而不是在执行时确认的;所以,后续变量 num 无论如何改变都不影响 defer 的输出结果。

func demo3_1() {
    num := 0
    defer func(n int) {
        fmt.Println("defer:", n)
    }(num)
    // 等同 defer fmt.Println("defer:", num)

    for i := 0; i < 10; i++ {
        num++
    }

    fmt.Println(num)}//10//defer: 0
登录后复制

运行 demo3_2,为什么这里 defer 的最终输出的结果会和变量 num 相同?因为这里使用的是指针。
defer 声明时,已经确认了形参p指针的指向地址,指向变量 num;后续变量 num 发生改变。所以在 defer 执行时,输出的是p指针指向的变量num的当前值。

func demo3_2() {
    num := 0
    p := &num    defer func(p *int) {
        fmt.Println("defer:", *p)
    }(p)

    for i := 0; i < 10; i++ {
        num++
    }

    fmt.Println(*p)}//10//defer: 10
登录后复制

再看一下 demo3_3,defer 打印的变量并没有通过函数参数传入,在defer执行时,才获取的”全局变量”num,所以 defer 输出结果与变量num一致。

func demo3_3() {
    num := 0
    defer func() {
        fmt.Println("defer:", num)
    }()

    for i := 0; i < 10; i++ {
        num++
    }

    fmt.Println(num)}//10//defer: 10
登录后复制

特性4:return 与 defer 执行顺序:return 先 defer 后

运行 demo4_1,可以发现 defer、return 都是在函数最后执行,但 return 先于 defer 执行;

func demo4_1() (int, error) {
    defer fmt.Println("defer")
    return fmt.Println("return")}// return// defer
登录后复制

这一点从输出结果上显而易见,但当 return、defer 的执行顺序和**函数返回值**“相遇”时,又将会产生许多复杂的场景。
在 demo4_2 中,函数使用命名返回值,最终输出结果为7。其中经历了这几个过程:

  1. (首先)变量 num 作为返回值,初始值为0;

  2. (其次)随后变量 num 被赋值为 10;

  3. (然后)return 时,变量 num 作为返回值被重新赋值为 2;

  4. (接着)defer 在 return 后执行,拿到变量 num 进行修改,值为7;

  5. (最后)变量 num 作为返回值,最终函数返回结果为7;

    func demo4_2() (num int) {
     num = 10
     defer func() {
         num += 5
     }()
    
     return 2}// 7
    登录后复制

再来看一个例子。
在 demo4_3 中,函数使用匿名返回值,最终结果输出为2。其中经历的过程是这样的:

  1. 进入函数,此时返回值变量并未创建;

  2. 创建变量 num,赋值为 10;

  3. return 时创建函数返回值变量,并赋值为2;这个返回值变量你可以把它看成匿名变量,或者是变量 a、b、c、d……,但它就不是变量 num;

  4. defer 时,无论怎样修改变量 num,都与函数返回值无关;

  5. 所以,最终的函数返回结果为2;

    func demo4_3() int {
     num := 10
     defer func() {
         num += 5
     }()
    
     return 2}// 2
    登录后复制

    特性5:发生 panic 时,已声明的 defer 会出栈执行

    运行 demo5_1,可以看到当出现 panic 时,会触发已经声明的 defer 出栈执行,随后在再 panic,而在 panic 之后声明的 defer 将得不到执行。

    func demo5_1() {
     defer fmt.Println(1)
     defer fmt.Println(2)
     defer fmt.Println(3)
    
     panic("没点赞异常") // 触发defer出栈执行
    
     defer fmt.Println(4) // 得不到执行}
    登录后复制

    正是利用这个特性,在 defer 中可以通过 recover 捕获 panic,防止程序崩溃。

    func demo5_2() {
     defer func() {
         if err := recover(); err != nil {
             fmt.Println(err, "问题不大")
         }
     }()
    
     panic("没点赞异常") // 触发defer出栈执行
    
     // ...}
    登录后复制

    完整代码:github.com/newbugcoder/learngo/tre...

以上是用八个demo搞懂Go语言defer的五大特性的详细内容。更多信息请关注PHP中文网其他相关文章!

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

AI Hentai Generator

AI Hentai Generator

免费生成ai无尽的。

热门文章

R.E.P.O.能量晶体解释及其做什么(黄色晶体)
2 周前 By 尊渡假赌尊渡假赌尊渡假赌
仓库:如何复兴队友
1 个月前 By 尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island冒险:如何获得巨型种子
4 周前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

Go的爬虫Colly中Queue线程的问题是什么? Go的爬虫Colly中Queue线程的问题是什么? Apr 02, 2025 pm 02:09 PM

Go爬虫Colly中的Queue线程问题探讨在使用Go语言的Colly爬虫库时,开发者常常会遇到关于线程和请求队列的问题。�...

Go语言中用于浮点数运算的库有哪些? Go语言中用于浮点数运算的库有哪些? Apr 02, 2025 pm 02:06 PM

Go语言中用于浮点数运算的库介绍在Go语言(也称为Golang)中,进行浮点数的加减乘除运算时,如何确保精度是�...

Go语言中哪些库是由大公司开发或知名的开源项目提供的? Go语言中哪些库是由大公司开发或知名的开源项目提供的? Apr 02, 2025 pm 04:12 PM

Go语言中哪些库是大公司开发或知名开源项目?在使用Go语言进行编程时,开发者常常会遇到一些常见的需求,�...

在使用Go语言和viper库时,为什么传递指针的指针是必要的? 在使用Go语言和viper库时,为什么传递指针的指针是必要的? Apr 02, 2025 pm 04:00 PM

Go指针语法及viper库使用中的寻址问题在使用Go语言进行编程时,理解指针的语法和使用方法至关重要,尤其是在...

如何解决Goland中自定义结构体标签不生效的问题? 如何解决Goland中自定义结构体标签不生效的问题? Apr 02, 2025 pm 12:51 PM

关于Goland中自定义结构体标签的问题在使用Goland进行Go语言开发时,经常会遇到一些配置上的问题。其中一个常�...

在 Go 语言中,为什么使用 Println 和 string() 函数打印字符串会出现不同的效果? 在 Go 语言中,为什么使用 Println 和 string() 函数打印字符串会出现不同的效果? Apr 02, 2025 pm 02:03 PM

Go语言中字符串打印的区别:使用Println与string()函数的效果差异在Go...

Go语言切片:单元素切片索引1截取为何不报错? Go语言切片:单元素切片索引1截取为何不报错? Apr 02, 2025 pm 02:24 PM

Go语言切片索引:为何单元素切片从索引1截取不报错?在Go语言中,切片是一种灵活的数据结构,它可以引用底�...

为什么Go语言中使用for range遍历slice并存入map时,所有值会变成最后一个元素? 为什么Go语言中使用for range遍历slice并存入map时,所有值会变成最后一个元素? Apr 02, 2025 pm 04:09 PM

为什么Go语言中的map迭代会导致所有值变成最后一个元素?在Go语言中,面对一些面试题时,经常会遇到关于map�...

See all articles