首页 > 文章列表 > Go 是否提供了处理单个语句中错误的标准或实际的内联方法?

Go 是否提供了处理单个语句中错误的标准或实际的内联方法?

389 2024-02-03
问题内容

在 Rust 中, Result 类型可以被认为是 Go 中的 (val, err) 模式,可以在它出现的语句中立即“展开”,并且不需要专用多行的复杂性到这个任务。例如,假设我们有一个函数 foo 返回一个数字或一个错误。在 Rust 中,处理这个错误(不忽略它)可以很简单:

let x = foo().unwrap() + 1;

在 Go 中,推荐的方法似乎是:

x, err := Foo()
if err != nil {
    panic(err)
}
x = x + 1

Go 是否提供了一种标准方法来像 Rust 一样在单个语句中立即处理错误?或者我应该坚持使用自己开发的解决方案?

我的临时解决方案是:

func unwrap[T any](val T, err error) T {
    if err != nil {
        panic(err)
    } else {
        return val
    }
}
<小时/>

我对过度节制感到有点惊讶。链接的五个问题都不是重复的,也没有一个具体回答我的问题。有些甚至看起来完全无关。我认为这是因为我的标题措辞不当。我已将标题改写为更符合我的实际问题。值得庆幸的是,u/norbjd 充分回答了我的问题。


正确答案


正如 @mkopriva 在评论中所说,Go 中没有内置的方法可以做到这一点。但是,您可以在 Must 名称下的许多库中找到此模式:

Go 上有一个公开提案,添加通用的 must.Do 方法 ,但还没有具体的内容。

目前推荐的方法是在代码中使用泛型定义 Must 方法(自 Go 1.18 起),就像您所做的那样:

func Must[T any](v T, err error) T {
    if err != nil {
        panic(err)
    }
    return v
}

并像这样使用它:

import (
    "errors"
    "fmt"
)

func Foo() (int, error) {
    return 1, nil
}

func Bar() (int, error) {
    return 0, errors.New("some error")
}

func main() {
    foo := Must(Foo())
    fmt.Println(foo)
    bar := Must(Bar()) // this will panic with "some error"
    fmt.Println(bar)
}

有人尝试创建库来提供此方法,例如 wingyplus/must,但作为回购协议指出,基于 Go 哲学:

因此,您可以在每个项目中定义自己的方法,或者使用这些类型的库,直到 Must 集成到 Go 标准库中(如果它曾经集成过)。