GO 中的 defer 有哪些注意事项?下

简介: 上次一起写了 3 个案例,咱们这一次继续,这一次的会比上一次的稍微不太一样

上次一起写了 3 个案例,咱们这一次继续,这一次的会比上一次的稍微不太一样

案例 1

还有一个也非常常用的案例,使用 defer 来捕获异常 ,也就是当程序崩溃的时候,defer 语句可以帮我们兜底,可以捕获异常后按照我们期望的逻辑进行执行,让程序回到正确的轨道上面

  • 一般使程序崩溃很简单, C/C++ 的时候,我们可以造异常,例如除 0 ,或者是数组越界等等就会导致程序崩溃
  • GO 里面造异常也可以按照上面这种方式,但是我们也可以使用 panic 函数来实现程序崩溃
  • 写一个简单的例子,把自己的程序搞崩
func testDefer() {
  defer func() {
    fmt.Println(" panic 前 ")
  }()
  panic("i panic") 
  defer func() {
    fmt.Println(" panic 后")
  }()
}
func main() {
  testDefer()
  fmt.Println("program over")
}

其实我们看 goland 工具就可以看到,下面这一句话是不会被执行的

defer func() { fmt.Println(" panic 后") }()

image.png

实际运行程序后,结果如下:

image.png

我们可以看到实际效果,程序是崩溃了,因为 fmt.Println("program over") 没有打印出来,且看图片,是有具体的 panic 信息的

从上述可以看明白, panic 之后的程序是不会执行的, panic 之前的 defer 语句会执行,因为他先入栈了

那么我们来捕获一下异常

还是上面的代码,我们来捕获一下异常,就是加一句话就可以了

func testDefer() {
  defer func() {
    fmt.Println(" panic 前 ")
    if err := recover(); err != nil {
      fmt.Println(" xdm , 我捕获到异常了,程序不用崩溃了 ")
    }
  }()
  panic("i panic") //触发defer出栈
  defer func() {
    fmt.Println(" panic 后")
  }()
}
func main() {
  testDefer()
  fmt.Println("program over")
}

看上述代码,panic 之后的程序仍然是不会执行的,但是我们加入了 recover() 语句, 他会帮助我们捕获异常,处理异常

image.png

执行上述代码后,效果如下:

> go run main.go
 panic 前
 xdm , 我捕获到异常了,程序不用崩溃了
program over

根据打印我们可以看出,结论和上面一个例子结论一致,并且加入异常捕获之后,程序的崩溃堆栈信息是不会打印出来了, 并且程序是正常退出了,我们可以看到正常打印了  program over

案例 2

咱们再来字后一个 案例 玩一下

上面都是出现 1 个 panic 的情况 ,那么,如果是出现多个 panic 是怎么玩的呢, defer 他还能捕获到 panic 信息吗?若出现 2 个 panic, defer 也能捕获 2 个吗?

func main() {
  defer func() {
    if err := recover(); err != nil {
      fmt.Println("xdm, 我捕获到异常了")
    }
  }()
  defer func() {
    panic("defer func  panic")
  }()
  panic(" i panic")
  panic(" last  panic")
  fmt.Println("baibai")
}

看到上述代码,一时间知道怎么走,但是我们看 goland 就知道 panic 后面的语句肯定不会执行

image.png

尝试分析一下,前面有说,defer 是按照栈的顺序来的

  • 第 1 个 derfer 先入栈
  • 第 2 个 defer 后入栈
  • 执行 panic 程序崩溃 应该会报错 i panic
  • 但是程序退出的时候,会出栈,会先执行 第 2 个 defer , 这个 defer 也是一个 panic, 则会覆盖刚才的 panic 信息,应该会报错 defer func  panic
  • 执行出栈,执行到第 1 个 defer ,异常被捕获了,程序正常退出,因此执行程序会有如下结果

shell

复制代码

xdm, 我捕获到异常了

好了,今天就到这里,感兴趣的朋友也可以玩起来


欢迎点赞,关注,收藏

朋友们,你的支持和鼓励,是我坚持分享,提高质量的动力

image.png

好了,本次就到这里

技术是开放的,我们的心态,更应是开放的。拥抱变化,向阳而生,努力向前行。

我是阿兵云原生,欢迎点赞关注收藏,下次见~

相关文章
|
11月前
|
Cloud Native Go
GO 中的 defer 有哪些注意事项?上
GO 中的 defer 有哪些注意事项?上
|
2月前
|
存储 Go
掌握 Go 语言的 defer 关键字
掌握 Go 语言的 defer 关键字
|
5月前
|
消息中间件 编译器 数据库连接
|
11月前
|
Cloud Native Go C++
GO 中的 defer 有哪些注意事项?下
GO 中的 defer 有哪些注意事项?下
|
11月前
|
存储 Java 编译器
GO 中 defer的实现原理
GO 中 defer的实现原理
|
存储 编译器 Go
Go的defer
在Go语言中,defer是一个关键字,用于延迟执行函数或方法的调用。当使用defer关键字时,被延迟执行的函数或方法会在包含它的函数返回之前被调用。
53 0
|
安全 Go
Go函数下篇:defer和闭包
Go函数下篇:defer和闭包
|
编译器 Go
【defer】1.12版本中的defer|Go主题月
defer在go中的作用就是延迟执行,一般常用来关闭一些io流等
97 0
|
Go
【defer】全新升级1.14以后的defer|Go主题月
在go1.14中,官方又对defer做了升级,据说这次升级把速度提升了一个量级。
106 0
|
存储 Go
【defer】升级版1.13中的defer|Go主题月
在1.13版本中,go语言改进了defer的创建。改为了直接用栈来存储_defer结构体。然后把栈上分配的_defer结构体注册到defer链表通过这样的方式避免在堆上分配_defer结构体。减少了堆和栈之间频繁复制导致的额外内存开销。
68 0