defer是什么?
在Go语言中,可以使用关键字defer向函数注册退出调用,即主函数退出时,defer后的函数才被调用。defer语句的作用是不管程序是否出现异常,均在函数退出时自动执行相关代码。
defer的用途
在函数中,程序员经常需要创建资源(比如:数据库连接、文件句柄、锁等) ,为了在函数执行完 毕后,及时的释放资源,Go 的设计者提供 defer (延时机制)。
package main import "fmt" //defer 可以向函数注册、退出调用,即当主调函数退出时,defer后的函数才被调用, // defer语句的作用就是不管程序是否出现异常,均在函数退出是自动执行相关代码 func sum(n1 int,n2 int) int{ //当执行到defer时,暂时不执行,会将defer后面的语句压入到独立的(defer栈) //当函数执行完毕后,再从defer栈 安装先入后出的方式出栈执行 //defer 将语句放入到栈时,也会将相关的值拷贝同时入栈*** defer fmt.Println("ok1 n1=",n1)//defer3 ok3 n1 20 defer fmt.Println("ok2 n2=",n2) //defer2 ok2 n2 10 //n1 ++ //n2 ++ res := n1 + n2 fmt.Println("ok3 res=",res)//1.ok1 return res } func main() { res :=sum(20,10) fmt.Println("res=",res) }
执行结果
结果分析:
从执行结果来看,res=30 是最先开始执行的部分,接下来是第二个defer n2为10 最后执行的是第一个defer n1为20 ,即当执行到defer时,暂时不执行,会将defer后面的语句压入到独立的(defer栈),当函数执行完毕后,再从defer栈,按照(先入后出)的方式出栈执行。
注意事项:
1) 当 go 执行到一个 defer 时,不会立即执行 defer 后的语句,而是将 defer 后的语句压入到一个栈 中[我为了讲课方便,暂时称该栈为 defer 栈], 然后继续执行函数下一个语句。
2) 当函数执行完毕后,在从 defer 栈中,依次从栈顶取出语句执行(注:遵守栈 先入后出的机制),
3) 在 defer 将语句放入到栈时,也会将相关的值拷贝同时入栈
package main import "fmt" //defer 可以向函数注册、退出调用,即当主调函数退出时,defer后的函数才被调用, // defer语句的作用就是不管程序是否出现异常,均在函数退出是自动执行相关代码 func sum(n1 int,n2 int) int{ //当执行到defer时,暂时不执行,会将defer后面的语句压入到独立的(defer栈) //当函数执行完毕后,再从defer栈 安装先入后出的方式出栈执行 //defer 将语句放入到栈时,也会将相关的值拷贝同时入栈*** defer fmt.Println("ok1 n1=",n1)//defer3 ok3 n1 20 defer fmt.Println("ok2 n2=",n2) //defer2 ok2 n2 10 n1 ++ n2 ++ res := n1 + n2 fmt.Println("ok3 res=",res)//1.ok1 return res } func main() { res :=sum(20,10) fmt.Println("res=",res) }
执行结果:
从结果看出n1++ ,n2++两个的值未入相关defer的栈,即defer 将语句放入到栈时,也会将相关的值拷贝同时入栈。