Golang中使用defer的那些坑

defer概述


    defer用来声明一个延迟函数,把这个函数放入到一个栈上,当外部的包含方法return之前,返回参数到调用方法之前调用,也可以说是运行到最外层方法体时调用。我们经常用他来做一些资源的释放,比如关闭io操作。

    defer是golang的一个特色功能,被称为“延迟调用函数”。当外部函数返回后执行defer。类似于其他语言的 try… catch … finally… 中的finally,当然差别还是明显的。在使用defer之前我们应该多了解defer的特性,这样才能避免使用上的误区。
 

defer以下几个特性,使用时需要关注下。

  • 即时的参数传递
  • 调用os.Exit()时defer不会被执行
  • defer与return的先后顺序

即时的参数传递

定义defer时传入的参数,是作为拷贝传递的。
也就是说,如果原来的变量值发生变化,不会影响传给defer的参数。

例子如下:

package main

import (
        "fmt"
)


func main(){
        test()
}

func test() {
        a := 0
        defer func (i int) {
                fmt.Println("in defer i:", i)
        }(a)
        a += 1
        fmt.Println("a:", a)
}

输出结果:

a: 1
in defer i: 0

可以看到,即使变量a发生变化,延迟执行时变量的值仍然是0,与定义defer时传入的值一样。

调用os.Exit()时defer不会被执行

当发生panic时,defer会被执行,但是当调用os.Exit()方法退出程序时,defer并不会被执行。

package main

import (
        "fmt"
        "os"
)


func main(){
        fmt.Println("main start")
        test()
}

func test() () {
        defer func () {
                fmt.Println("in defer ... ")
        }()

        os.Exit(0)
}

输出结果:

main start

defer定义的内容没有输出。

defer 与 return先后顺序

先来看两个例子:一个是返回匿名变量,一个是返回命名变量。

返回匿名变量

package main

import (
        "fmt"
)


func main(){
        i := test()
        fmt.Println("main i:", i)
}

func test() int {
        a := 0
        defer func () {
                a = 2
        }()

        a = 1
        return a
}

定义a为0, 接着修改为1,最后在defer中将a修改为2。

在main中返回的值仍然是1.

输出结果:

main i: 1

返回命名变量

package main

import (
        "fmt"
)


func main(){
        i := test()
        fmt.Println("main i:", i)
}

func test() (a int) {
        defer func () {
                a = 2
        }()

        a = 1
        return a
}

defer中修改a为2,能够返回给调用方。

输出结果:

main i: 2

实际上,defer 函数的执行既不是在 return 之后也不是在 return 之前,而是 return 语句包含了对 defer 函数的调用,即 return 会被翻译成如下几条伪指令:

保存返回值到栈上(如果是匿名变量,需要定义变量并赋值)

调用defer函数(如果有defer函数,则调用并执行)

调整函数栈

retq指令返回(如果是匿名变量,直接返回新定义的变量,如果是命名变量,直接返回命名变量)

 命名变量返回时,不会创建新的变量,所以defer的修改会返回去。
而匿名变量,会创建新的变量,defer中的修改,还是修改原来的变量,所以修改不能返回去。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

金戈鐡馬

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值