美文网首页
defer 和闭包

defer 和闭包

作者: Robin92 | 来源:发表于2020-01-04 21:42 被阅读0次

今天看了一篇 文章,确实发现自己对 defer 的掌握还不够,文章出了这样一道题。问,以下三个函数,运行后返回是多少,为什么?

func increaseA() int {
    var i int
    defer func() {
        i++
    }()
    return i 
}

func increaseB() (r int) {
    defer func() {
        r++
    }()
    return r
}

func increaseC() (r int) {
    defer func(r int) {
        r++
    }(r)
    return r
}

这个题 increaseA() 返回 0,而 increaseB() 返回 1,increaseC() 返回 0。

defer 的作用

defer 是 Go 语言提供的一种用于注册延迟调用的机制,每一次 defer 都会把函数压入栈中,当前函数返回前再把延迟函数取出并执行。

defer 定义的函数会先进入一个栈,函数 return 前,会按先进后出(FILO)的顺序执行。也就是说最先被定义的 defer 语句最后执行。

踩坑点

但注意,这远远没概括完 defer 的所有特性。了解 defer 还要牢记以下几个特点。

使用 defer 最容易踩坑的地方是 函数返回参数上指定了参数名 时,返回结果有可能被修改。

defer 语句定义时,对 外部变量的引用 是有两种方式的,分别是作为 函数参数 和作为 闭包引用

  • 作为 函数参数,则在 defer 定义时 就把值传递给 defer,并被 缓存 起来;
  • 作为 闭包引用 的话,则会在 defer 函数真正调用时根据整个上下文确定当前的值。

规则一

实参在 defer 函数定义时就取值并缓存起来了。

如:

func trace(str string) string {
    fmt.Println("entering " + str)
    return str
}
func leave(str string) {
    fmt.Println("leaving " + str)
}
func point() {
    defer leave(trace("point"))
    fmt.Println("in point")
}
func main() {
    point()
}

输出

entering point
in point
leaving point

即,在 defer leave(trace("point")) 的流程中,会先执行 trace("point"),然后执行 函数其他语句,最后再来执行 leave(xxx) 函数。

规则二

要完全理解第二条规则,需要了解 returndefer 是怎么运行的。

函数内的 return xxx 并不是一个原子执行的返回:即不是先执行 return xxx 再执行 defer,也不是先执行 defer 再执行 return xxx。而是将 return xxx 拆分开来,经过编译后执行过程如下:

1. 返回变量 = xxx
2. 调用 defer 函数(有可能更新返回变量的值)
3. return 返回变量

这里 返回变量 有点特殊,如果在函数签名中定义了返回变量,那么就是它;而如果没有定义,那么返回变量会是一个匿名的变量。defer 函数能够更改函数返回值的情况,都是在函数签名中定义了返回变量的情景。

比如,在以下定义了函数返回变量的案例中,按上述方法拆解一下下面三个函数:

1.
func f1() (r int) {
    defer func() {
        r++
    }()
    return 0
}

2.
func f2() (r int) {
    t := 5
    defer func() {
        t = t + 5
    }()
    return t
}

3.
func f3() (r int) {
    defer func(r int) { // 作为函数参数传入 defer 函数
        r = r + 5 
    }(r)
    return 1
}

则,第一题中在执行 return 0 时,分成以下三步:

r = 0 // 1. 赋值
func() { // 2. 运行 defer 函数 r++,r = 1
    r++
}()
return r // 3. return,即返回结果为 1

第二题中的执行 return t 时,分成一下三步:

r = t (= 5) // 1. 赋值,r 取值 5
func() { // 2. 执行 defer 函数,执行后 t = 10,但 r = 5
    t = t + 5
}()
return r // 3. return r,即返回 5

第三题中执行 return 1 时,分为以下三步:

r = 1 // 1. 赋值, r 取值 1
func(r int) { // 2. 执行 defer 函数,但作为函数参数传入
    r = r + 5 // 执行后 r = 6 ,但这是局部变量,函数外仍是 1
}(r)
return r // 3. return r, 即返回 1

回到最初的三道题

将解释以注释的方式写在代码中:

func increaseA() int {
    var i int
    defer func() {
        i++
    }() 
    return i // 1. annoy = i (= 0); 2. 执行 defer 函数后 i = 1, 但 annoy = 0; 3. return annoy, 即 0
}

func increaseB() (r int) {
    defer func() {
        r++
    }()
    return r // 1. r = r (= 0); 2. 执行 defer 后 r = 1; 3. return r, 即返回 1
}

func increaseC() (r int) {
    defer func(r int) {
        r++
    }(r)
    return r // 1. r = 0; 2. 执行 defer r++, 但函数外 r = 0; 3. return r, 即返回 0
}

总结

关于 defer 函数牢记两条规则,并附加一个通用规则

  • defer 函数的 参数(实参),在 defer 定义时 就把值传递给 defer,并被 缓存 起来;
  • 作为 闭包引用 的话,则会在 defer 函数真正调用时根据整个上下文确定当前的值。这时要牢记“三步走”拆解上下文
  • 同时记得,传入函数的参数都是局部变量了(除非匿名函数或引用传入)

相关文章

  • defer 和闭包

    今天看了一篇 文章[https://mp.weixin.qq.com/s?__biz=MzI2MDA1MTcxMg...

  • go defer 的使用和陷阱

    前言 初学 go 的同学都应该了解 defer, defer 让人又爱又恨;当 defer 和 闭包结合在一起的时...

  • 【golang】底层易错点总结

    闭包特殊性 闭包对捕获的外部变量并不是传值方式访问,而是以引用的方式访问。如以下defer 因为defer后面是一...

  • go 基本函数

    多返回值函数 递归 函数类型 type 回调函数 和多态(强大) 匿名函数 与 闭包 关键字defer(defer...

  • golang关于defer的问题

    要弄明白defer首先得搞清楚闭包和局部作用域。 闭包 闭包是匿名函数与匿名函数所引用环境的组合。看一个例子: 看...

  • Golang 函数

    匿名函数 回调函数 闭包 defer 延迟执行函数 多个defer的时候 先延迟的最后执行 先进后出栈原理...

  • golang 中的闭包和defer

    golang中的defer和闭包对很多初学者来说,有时候有很多坑,但是很多介绍的文章有写的乱七八糟.放假了没事可干...

  • swift中的defer使用场景

    延迟 个人理解类似闭包,会延迟执行,但只要执行了defer定义的代码就一定会执行。 执行顺序 defer的位置越靠...

  • go 闭包与匿名函数这一篇就够了

    最近在看go语言中的defer,里面涉及到了闭包,之前只是对闭包有了解,但是并没有深入的研究过,这次就深入研究一下...

  • go优化——容易犯错点记载

    内容 1 切片与数组2 defer3 make与new4 方法与函数5 闭包6 循环 1 切片和数组 数组和结构体...

网友评论

      本文标题:defer 和闭包

      本文链接:https://www.haomeiwen.com/subject/qwefactx.html