关于Go关键字defer的一些坑

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/lengyuezuixue/article/details/78540955

    defer意为延迟,在Go语言中用于延迟执行一个函数。它可以帮助我们处理容易忽略的问题,如资源释放、连接关闭等。但在时间使用过程中,有一些需要注意的地方(坑)。

结论

1.若函数中有多个defer,其执行顺序为先进后出,可以理解为栈。

 

package main

import "fmt"

func main() {
  for i := 0; i < 5; i++ {
    defer fmt.Println(i)
  }
}

Output:
4
3
2
1
0

 

2.return会做几件事

 

    1.给返回值赋值;

     2.调用defer表达式;

     3.返回调用函数;

 

package main

import "fmt"

func main() {
    fmt.Println(increase(1))
}

func increase(d int) (ret int) {
  defer func() {
    ret++
  }()

  return d
}
  
Output:
2


2.若defer表达式有返回值,将会被丢弃。

 

 

闭包和匿名函数

匿名函数:没有函数名的函数。

闭包:可以使用另外一个函数作用域中的变量的函数。

 

在实际开发中,defer的使用经常伴随着闭包和匿名函数的使用。小心踩坑:

 

package main

import "fmt"

func main() {
    for i := 0; i < 5; i++ {
        defer func() {
            fmt.Println(i)
        }()
    }
}

Output:
5
5
5
5
5

解释一下,defer表达式中的i是对for循环中i的引用。到最后,i加到5,故而最后全部打印5。

 

如果将i作为参数传入defer表达式,在传入最初就会进行求值保存,只是没有执行延迟函数而已。

 

for i := 0; i < 5; i++ {
    defer func(idx int) {
        fmt.Println(idx)
    }(i) // 传入的 i,会立即被求值保存为 idx
}


巩固一下

 

 

 

         为了巩固一下上面的知识点,我们来思考几个例子。

例1:

 

func f() (result int) {
    defer func() {
        result++
    }()
    return 0
}


例2:

 

 

func f() (r int) {
    t := 5
    defer func() {
        t = t + 5
    }()
    return t
}

例3:

func f() (r int) {
    defer func(r int) {
        r = r + 5
    }(r)
    return 1
}

 

 

 

分析如下:

    例1,比较简单,参考结论2,将0赋值为result,defer延迟函数修改result,最后返回给调用函数,正确答案是1。

    例2,defer是在t赋值给r之后执行的,而defer延迟函数只是改变了t的值,r不变,正确答案是5。

    例3,这里将r作为参数传入了defer表达式。故而func(r int)中的r非func f()(r int)中的r,只是参数命名相同而已。正确答案是1。

 

本文参与:https://www.yuque.com/docs/share/d5cc0cb1-276e-4d78-a5bf-139a07c0c2e8

 

展开阅读全文

没有更多推荐了,返回首页