golang recover函数使用中的一些坑解析

 更新时间:2023年03月19日 11:22:07   作者:机器马喝啤酒  
这篇文章主要为大家介绍了golang recover函数使用中的一些坑解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

正文

众所周知golang 中recover函数可以捕捉panic,防止在出现异常的情况下服务整个不可用。然而某些情况下recover也无法catch panic。下面就会说一些这些情况。

一,正常情况下

package main
import "fmt"
func main(){
    defer func(){
        if err := recover();err != nil{
            fmt.Printf("err = %v",err)
        }
    }()
    panic("a panic")
}
打印结果:
err = a panic
Process finished with exit code 0

能正常catch panic

二, goroutine中panic 

之前线上环境出现过接口出现panic导致服务不可用的情况,于是同事就直接在main函数加了个recover认为万事无忧了。实际上recover并不能捕捉到协程中的panic。

package main
import "fmt"
func main(){
    defer func(){
        if err := recover();err != nil{
            fmt.Printf("err = %v",err)
        }
    }()
    go func(){
        panic("a panic")
    }()
    select{}
}
打印结果:
panic: a panic
goroutine 6 [running]:
main.main.func2()
    I:/goProject/catchPanic.go:13 +0x40
created by main.main
    I:/goProject/catchPanic.go:12 +0x5e

实际上还是会panic导致服务不可用。

正确写法

package main
import "fmt"
func main(){
    go func(){
        defer func(){
            if err := recover();err != nil{
                fmt.Printf("err = %v",err)
            }
        }()
        panic("a panic")
    }()
    select {}
}
返回值:
fatal error: all goroutines are asleep - deadlock!
goroutine 1 [select (no cases)]:
main.main()
    I:/goProject/catchPanic.go:15 +0x41
err = a panic
Process finished with exit code 2

可以看到panic被正常捕捉,同时因为select语句陷入阻塞,报了一个死锁的错。

三,间接调用recover

在我想要把recover封装成成一个函数的时候,发现recover并没有生效,因为recover只有在被defer语句直接调用的时候才会生效。当recover在其他函数内部的时候无法正确捕捉到panic。

package main
import "fmt"
func main(){
    defer cover()
    panic("a panic")
}
func cover(){
    defer func(){
        if err := recover();err!= nil{
            fmt.Println(err)
        }
    }()
}
返回值:
panic: a panic
goroutine 1 [running]:
main.main()
    I:/goProject/catchPanic.go:7 +0x62

四,nil panic

panic要被捕捉,还需要满足一种条件,就是panic不是nil panic,否则在进行捕获判断的时候无法知道是panic没有发生还是panic本身就是nil。

例如以下代码

package main
import "fmt"
func main() {
    defer func(){
        if err := recover();err != nil{
            fmt.Println(err)
        }
        fmt.Println("after recover")
    }()
    panic(nil)
    select{}
}
返回值:
after recover

recover并没有正确处理异常,因为异常的值为nil。

五,总结

这篇文章讲述了三种recover会失效的情况。

  •  携程中出现panic
  • defer不直接调用recover
  • panic的值为nil值

写代码的时候需要注意避免因为这几种情况的出现而导致服务不可用。以上就是golang新手常遇见的一些坑。

以上就是golang recover函数使用中的一些坑解析的详细内容,更多关于golang recover函数坑的资料请关注脚本之家其它相关文章!

相关文章

  • 详解Go中Set的实现方式

    详解Go中Set的实现方式

    这篇文章主要介绍了详解Go中Set的实现方式,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-09-09
  • Go字符串切片操作str1[:index]的使用

    Go字符串切片操作str1[:index]的使用

    Go字符串切片str1[:index]从起始位置0到index-1截取,不复制数据,利用字符串不可变性和共享内存机制提升性能,具有一定的参考价值,感兴趣的可以了解一下
    2025-06-06
  • go自动下载所有的依赖包go module使用详解

    go自动下载所有的依赖包go module使用详解

    这篇文章主要介绍了go自动下载所有的依赖包go module使用详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-03-03
  • golang 随机数的两种方式

    golang 随机数的两种方式

    本文主要介绍了golang 随机数的两种方式,一种是伪随机,另一种是真随机,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-07-07
  • go语言数据类型之字符串string

    go语言数据类型之字符串string

    这篇文章介绍了go语言数据类型之字符串string,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-07-07
  • Golang 并发读写锁的具体实现

    Golang 并发读写锁的具体实现

    Go语言中的sync.RWMutex提供了读写锁机制,允许多个协程并发读取共享资源,但在写操作时保持独占性,本文主要介绍了Golang 并发读写锁的具体实现,感兴趣的可以了解一下
    2025-02-02
  • golang gc的内部优化详细介绍

    golang gc的内部优化详细介绍

    Go编译器在垃圾回收(GC)的扫描标记阶段,对存储无指针键值对的map进行了优化,即在GC扫描时不深入扫描map内部数据,只检查map本身是否需要回收,这一优化显著提升了GC扫描的速度,从而减少了GC对程序性能的影响
    2024-10-10
  • Go语言学习之Switch语句的使用

    Go语言学习之Switch语句的使用

    这篇文章主要通过一些示例为大家介绍一下Go语言中Switch语句的基本语法以及使用,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一下
    2022-06-06
  • 详解go语言中并发安全和锁问题

    详解go语言中并发安全和锁问题

    这篇文章主要介绍了go语言中并发安全和锁问题,包含互斥锁解锁过程,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-10-10
  • Golang websocket协议使用浅析

    Golang websocket协议使用浅析

    这篇文章主要介绍了Golang websocket协议的使用,WebSocket是一种新型的网络通信协议,可以在Web应用程序中实现双向通信,感兴趣想要详细了解可以参考下文
    2023-05-05

最新评论