美文网首页
golang中的defer关键字

golang中的defer关键字

作者: n_xy | 来源:发表于2021-03-27 15:16 被阅读0次

defer关键字将函数调用计划到外层函数返回时执行,典型的用法就是互斥锁释放,文件关闭。这样的函数是不会因为随着返回分支的数量变多而忘记关闭。
但是defer的执行特定如果没有了解的话,很容易掉坑里。

多个defer的执行顺序

首先,对于下述例子,输出为何

for i := 0; i < 5; i++ {
    defer fmt.Printf("%d ", i)
}

首先说明,函数存在多个defer语句,调用的时的顺序为FILO,可以想想成压栈处理,即吧defer的调用先存入栈中,然后调用时在一个个出栈执行.
明白这个,答案就很明白了
4 3 2 1 0

带参defer的问题

考虑下述程序执行

func trace(s string) string {
    fmt.Println("entering:", s)
    return s
}

func un(s string) {
    fmt.Println("leaving:", s)
}

func a() {
    defer un(trace("a"))
    fmt.Println("in a")
}

func b() {
    defer un(trace("b"))
    fmt.Println("in b")
    a()
}

func main() {
    b()
}

写下答案,
标准答案为

entering: b
in b
entering: a
in a
leaving: a
leaving: b

道理其实很简单(知道了之后~~),defer语句调用的函数的参数不是在调用时才确定,而是在程序执行到这条语句是就确定了。
所以上述函数b中执行到 defer un(trace("b"))时,会首先确定un函数的参数,即执行trace("b")

一些例子

func b(t int ) (r int){

    defer func() {t=t+1}()

    return t
}

func main() {
    fmt.Println(b(1))
}
output// 1

这个需要知道,go中return语句不是原子操作,可以理解为两步

  1. 赋值
    在这个例子中,返回值指定了r,所以第一步 r=t
  2. 返回
    return r

关键在于defer的执行时机,他是在这两步中间执行的。

这个是由开篇的例子引出
具体

for i := 0; i < 5; i++ {
    defer func(){ fmt.Printf("%d ", i)}
}
output// 55555
for i := 0; i < 5; i++ {
    defer func(n int){ fmt.Printf("%d ", n)(i)}
}
output//01234

经典闭包问题,go中函数调用都是值拷贝,for循环中i始终是一个地址,所以联系到上边说的,defer函数调用参数实在语句执行时候确定的,所以传入的始终是i的地址的值。类似还有结构体的for-range

type field struct {
 name string
}
 
func (p *field) print() {
 fmt.Println(p.name)
}
 
func main() {
 data := []field{{"one"},{"two"},{"three"}}
 1/////
   for _,v := range data {
        defer v.print()
   } //goroutines print: three, three, three


2
  for _,v := range data {
        v := v
        defer v.print()
  }three two one
 
 time.Sleep(3 * time.Second)
}

相关文章

  • go defer理解

    golang的defer是怎么工作的?defer在golang里是一个很基础的关键字,在函数内部使用defer声明...

  • GO中defer关键字

    defer关键字在go中用于return之前执行,在golang官方文档中有说明。defer的用法类似java中的...

  • Golang之Defer

    引用 golang defer实现原理 Golang之轻松化解defer的温柔陷阱 Golang中defer、re...

  • Golang-defer关键字

    defer关键字 defer是Golang中一个非常重要的关键字,主要是用于注册延迟调用,这些调用在return时...

  • 2017-12-04

    Golang,Panic,Defer,Recover 在golang中,recover在defer里发挥作用。 一...

  • golang中的defer关键字

    defer关键字将函数调用计划到外层函数返回时执行,典型的用法就是互斥锁释放,文件关闭。这样的函数是不会因为随着返...

  • 12 Golang defer panic recover

    defer Golang中的defer会将其后面跟随的语句进行延迟处理。在defer归属的函数即将返回时,将延迟处...

  • golang 中的 defer

    defer 是在return 之前执行的但是return xxx 并不是一条原始指令

  • Go语言defer关键字

    Go语言defer关键字 defer关键字用于延缓函数的执行 只需要在调用普通函数或方法前加上关键字defer,就...

  • golang中defer执行时机与常见问题

    defer是什么? defer的用途 defer的执行时机 了解defer的执行时机首先要知道golang的ret...

网友评论

      本文标题:golang中的defer关键字

      本文链接:https://www.haomeiwen.com/subject/kaqyhltx.html