defer是什么?

在Go语言中,可以使用关键字defer向函数注册退出调用,即主函数退出时,defer后的函数才被调用。defer语句的作用是不管程序是否出现异常,均在函数退出时自动执行相关代码。

defer的用途

在函数中,程序员经常需要创建资源(比如:数据库连接、文件句柄、锁等) ,为了在函数执行完毕后,及时的释放资源,Go 的设计者提供 defer (延时机制)。

packagemainimport"fmt"//defer可以向函数注册、退出调用,即当主调函数退出时,defer后的函数才被调用,//defer语句的作用就是不管程序是否出现异常,均在函数退出是自动执行相关代码funcsum(n1int,n2int)int{//当执行到defer时,暂时不执行,会将defer后面的语句压入到独立的(defer栈)//当函数执行完毕后,再从defer栈安装先入后出的方式出栈执行//defer将语句放入到栈时,也会将相关的值拷贝同时入栈***deferfmt.Println("ok1n1=",n1)//defer3ok3n120deferfmt.Println("ok2n2=",n2)//defer2ok2n210//n1++//n2++res:=n1+n2fmt.Println("ok3res=",res)//1.ok1returnres}funcmain(){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 将语句放入到栈时,也会将相关的值拷贝同时入栈

packagemainimport"fmt"//defer可以向函数注册、退出调用,即当主调函数退出时,defer后的函数才被调用,//defer语句的作用就是不管程序是否出现异常,均在函数退出是自动执行相关代码funcsum(n1int,n2int)int{//当执行到defer时,暂时不执行,会将defer后面的语句压入到独立的(defer栈)//当函数执行完毕后,再从defer栈安装先入后出的方式出栈执行//defer将语句放入到栈时,也会将相关的值拷贝同时入栈***deferfmt.Println("ok1n1=",n1)//defer3ok3n120deferfmt.Println("ok2n2=",n2)//defer2ok2n210n1++n2++res:=n1+n2fmt.Println("ok3res=",res)//1.ok1returnres}funcmain(){res:=sum(20,10)fmt.Println("res=",res)}

执行结果:


从结果看出n1++ ,n2++两个的值未入相关defer的栈,即defer将语句放入到栈时,也会将相关的值拷贝同时入栈。