defer 延时机制

在函数执行过后执行。
作用:在函数执行过后及时释放资源。
最佳的实践在于,做一些执行后的操作,释放资源,比如:数据库连接、文件句柄、锁等。

在 golang 中的通常做法是,创建资源后,立即使用 defer 进行释放。

1.基本用法

原理是使用了 defer 关键字后,会将修饰语句加入到一个栈中,这里称为 defer 栈中,直到程序执行后,最后执行,当函数执行完毕后,在从 defer 栈中,依次从栈顶取出语句执行,先入后出的机制。
defer对执行流程的改变,只是在本函数内有用。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
package main

import "fmt"

func sum(n1 int, n2 int) int {
defer fmt.Printf("n1=%d, n2=%d\n",n1, n2)
res := n1 + n2
defer fmt.Printf("res1=%d\n", res)
return res
}

func main() {
res := sum(10, 20)
fmt.Printf("res2=%d", res)
}

结果

res1=30
n1=10, n2=20
res2=30

2.注意事项

defer 拷贝入栈后,值是拷贝进去的,后续的操作,不影响已拷贝的值。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
package main

import "fmt"

func sum2(n1 int, n2 int) int {
defer fmt.Println("n1=", n1)
defer fmt.Println("n2=", n2)
n1++
n2++
res := n1 + n2
defer fmt.Printf("res1=%d\n", res)
return res
}


func main() {
res2 := sum2(10, 20)
fmt.Printf("res2=%d", res2)
}

结果

res1=32
n2= 20
n1= 10
res2=32

3.defer 的意义

在于可以及时的释放函数创建的资源。很多时间在开发中,打开了资源后,就忘了关闭,defer 的机制可以让程序员在开发时,使用资源后立即加上关闭操作,也不影响后续的代码执行。类似于 java 的finally的作用,不同在于语法上defer可以写在前面。