在GO语言中使用defer函数获取panic()参数

时间:2013-08-15 15:08:02

标签: go

我有一个函数A调用函数B,它有时会根据无效数据调用panic。 在函数A defer函数中,我想知道消息函数B传递给panic(),以便我可以通过网络将json中的错误报告给客户端。

e.g。

func A( abc data) result string{
  defer func(){
    // get panic args and return result.
  }

  xx = B( abc[0] );
  yy = B( abc[1] );
  ...
}

功能B使用恐慌的原因是为了避免非常大量的

err := B(abc)
if err != nil {
    ...
}

在函数A中,使代码更易于阅读和维护。

2 个答案:

答案 0 :(得分:4)

例如:

package main

import (
        "errors"
        "fmt"
)

func A(s string) (result string, err error) {
        defer func() {
                if e := recover(); e != nil {
                        switch x := e.(type) {
                        case error:
                                err = x
                        default:
                                err = fmt.Errorf("%v", x)
                        }
                }
        }()

        B(s)
        return "returned", nil
}

func B(s string) {
        switch s {
        case "ok":
                return
        case "fail":
                panic(errors.New("failed"))
        case "fail miserably":
                panic(42)
        default:
                a, b := 1, 0
                if a/b != 0 {
                        panic("ouch")
                }
        }
}

func main() {
        s, err := A("ok")
        fmt.Printf("%q, %T(%#v)\n", s, err, err)

        s, err = A("fail")
        fmt.Printf("%q, %T(%#v)\n", s, err, err)

        s, err = A("fail miserably")
        fmt.Printf("%q, %T(%#v)\n", s, err, err)

        s, err = A("")
        fmt.Printf("%q, %T(%#v)\n", s, err, err)
}

Playground


输出:

"returned", <nil>(<nil>)
"", *errors.errorString(&errors.errorString{s:"failed"})
"", *errors.errorString(&errors.errorString{s:"42"})
"", runtime.errorString("integer divide by zero")

答案 1 :(得分:0)

你想要的是recover功能。你想推迟它是正确的 - 恢复只能在延迟函数中正常工作(如果你在体内调用它,如果没有恐慌就会返回nil,或者在发生恐慌时跳过它)。 Recover返回在空接口中惊慌失措的值:

func A(abc data) result string {
    defer func() {
        p := recover() // p is an interface{} value, and will be nil if there was no panic
    }() // You have to call the function
    ...
}