当前位置 博文首页 > 试了下Golang实现try catch的方法

    试了下Golang实现try catch的方法

    作者:锐玩道 时间:2021-08-03 18:04

    目录
    • 前言
    • 能不能 try catch
    • 加入处理异常方法
    • 总结

    前言

    Golang语言有诸多优点:静态编译、协程、堪比c语言的高性能。

    但是也有一些令人发指的地方 —— 经常被人调侃 五行代码,三行错误处理 的异常错误处理方式,如下问题出现的地方

    func readFile(path string) ([]byte, error) {
        file, err := os.Open(path)
        if err != nil {
            return nil, err
        }
        defer file.Close()
        return ioutil.ReadAll(file)
    }
    

    函数readFile的功能是读出指定文件本身的内容并将其返回,同时当有错误发生时立即向调用方报告。
    根据Golang编程范式,每个可能导致错误的函数都应该将error作为最后一个返回值,所以Golang代码中随处都是"if err != nil"语句。 五行代码,三行错误处理 的调侃段子由此在圈内流行

    能不能 try catch

    我之前常用编程语言 Python 和 PHP 的时候,早已经习惯 try catch 处理异常的方式。于是对 Golang 使用流程控制 配合 panic()、recover() 一直耿耿于怀,为了以后舒服今天就来实现一下原来 try catch异常错误处理方式

    Golang 语言中拥有 recover() 函数作为宕机恢复机制,让程序在宕机流程中的 goroutine 中恢复。

    所以我们能够用 recover() 函数作为程序中是否出现的钩子。当出现异常时,由它调起我们的异常处理函数。首先简单实现一下:

    package main
    
    import "fmt"
    
    func try(userFn func()) {
     defer func() {
      if err := recover();err != nil{
       fmt.Printf("程序执行发生异常: %v\n", err)
      }
     }()
    
     userFn()
    }
    
    func foo(num int)  {
     if num < 10 {
      panic("number is less than 10")
     }else {
      panic("number is greater than 10")
     }
    }
    
    func main()  {
     try(func() {
      foo(9)
     })
    
     try(func() {
      foo(11)
     })
    }
    
    

    go run main.go 运行:

    $ go run _test/demo3/test3.go
    程序执行发生异常: number is less than 10
    程序执行发生异常: number is greater than 10

    从上可以看出,recover() 函数完成我们让它作为钩子,的作用,调起了打印 程序执行发生异常 的信息。

    加入处理异常方法

    捕抓到异常还需要处理异常,而不是简单打印 程序执行发生异常。所以还需要增加匿名方法作为异常处理方法,改造如下:

    func try(userFn func(), catchFn func(err interface{})) {
     defer func() {
      if err := recover();err != nil{
       catchFn(err)
      }
     }()
    
     userFn()
    }
    
    ...
    
    func main()  {
     try(func() {
      foo(9)
     }, func(err interface{}) {
      fmt.Printf("程序执行发生异常: %v\n", err)
     })
    
     try(func() {
      foo(11)
     }, func(err interface{}) {
      fmt.Printf("程序执行发生异常: %v\n", err)
     })
    }
    
    

    这样就可以随心所欲传入自定义处理异常的方法

    总结

    以上只是自己简单实现了try catch,其中还有很多缺陷,比如:不支持 finally 函数处理,try catch封装过于简单。

    jsjbwy