Golang函数中defer的作用以及细节

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)

}

执行结果

Golang函数中defer的作用以及细节

结果分析:

   从执行结果来看,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)

}

执行结果:

Golang函数中defer的作用以及细节

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


上一篇:defer和async的区别


下一篇:Go36-26-互斥锁与读写锁