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

结论

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

  1. package main
  2. import "fmt"
  3. func main() {
  4. for i := 0; i < 5; i++ {
  5. defer fmt.Println(i)
  6. }
  7. }

Output:

  1. 4
  2. 3
  3. 2
  4. 1
  5. 0

2.return会做几件事

  1. 给返回值赋值;

  2. 调用defer表达式;

  3. 返回调用函数;

  1. package main
  2. import "fmt"
  3. func main() {
  4. fmt.Println(increase(1))
  5. }
  6. func increase(d int) (ret int) {
  7. defer func() {
  8. ret++
  9. }()
  10. return d
  11. }

Output:

  1. 2

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

闭包和匿名函数

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

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

  1. package main
  2. import "fmt"
  3. func main() {
  4. for i := 0; i < 5; i++ {
  5. defer func() {
  6. fmt.Println(i)
  7. }()
  8. }
  9. }

Output:

  1. 5
  2. 5
  3. 5
  4. 5
  5. 5

解释一下,defer表达式中的i是对for循环中i的引用。到最后,i加到5,故而最后全部打印5。
如果将i作为参数传入defer表达式,在传入最初就会进行求值保存,只是没有执行延迟函数而已。

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

巩固一下

  1. 为了巩固一下上面的知识点,我们来思考几个例子。<br />例1
  1. func f() (result int) {
  2. defer func() {
  3. result++
  4. }()
  5. return 0
  6. }

例2:

  1. func f() (r int) {
  2. t := 5
  3. defer func() {
  4. t = t + 5
  5. }()
  6. return t
  7. }

例3:

  1. func f() (r int) {
  2. defer func(r int) {
  3. r = r + 5
  4. }(r)
  5. return 1
  6. }

分析如下:

  1. 1,比较简单,参考结论2,将0赋值为resultdefer延迟函数修改result,最后返回给调用函数,正确答案是1。<br /> 2defer是在t赋值给r之后执行的,而defer延迟函数只是改变了t的值,r不变,正确答案是5。<br /> 3,这里将r作为参数传入了defer表达式。故而func(r int)中的rfunc f()(r int)中的r,只是参数命名相同而已。正确答案是1

关于Go关键字defer的一些坑 - 图1