golang error处理之多error聚合

下面这段代码取自于云原生开源项目karmada,我们可以借鉴以下它的检查参数中处理error的方法。

// Validate checks Options and return a slice of found errs.
func (o *Options) Validate() field.ErrorList {
    errs := field.ErrorList{}
    newPath := field.NewPath("Options")
​
    skippedResourceConfig := util.NewSkippedResourceConfig()
    if err := skippedResourceConfig.Parse(o.SkippedPropagatingAPIs); err != nil {
        errs = append(errs, field.Invalid(newPath.Child("SkippedPropagatingAPIs"), o.SkippedPropagatingAPIs, "Invalid API string"))
    }
    if o.SecurePort < 0 || o.SecurePort > 65535 {
        errs = append(errs, field.Invalid(newPath.Child("SecurePort"), o.SecurePort, "must be between 0 and 65535 inclusive"))
    }
    if o.ClusterStatusUpdateFrequency.Duration <= 0 {
        errs = append(errs, field.Invalid(newPath.Child("ClusterStatusUpdateFrequency"), o.ClusterStatusUpdateFrequency, "must be greater than 0"))
    }
    if o.ClusterLeaseDuration.Duration <= 0 {
        errs = append(errs, field.Invalid(newPath.Child("ClusterLeaseDuration"), o.ClusterLeaseDuration, "must be greater than 0"))
    }
    if o.ClusterMonitorPeriod.Duration <= 0 {
        errs = append(errs, field.Invalid(newPath.Child("ClusterMonitorPeriod"), o.ClusterMonitorPeriod, "must be greater than 0"))
    }
    if o.ClusterMonitorGracePeriod.Duration <= 0 {
        errs = append(errs, field.Invalid(newPath.Child("ClusterMonitorGracePeriod"), o.ClusterMonitorGracePeriod, "must be greater than 0"))
    }
    if o.ClusterStartupGracePeriod.Duration <= 0 {
        errs = append(errs, field.Invalid(newPath.Child("ClusterStartupGracePeriod"), o.ClusterStartupGracePeriod, "must be greater than 0"))
    }
​
    return errs
}
​

因为要检查的参数有多个,所以在检查的过程中遇到不合理的配置参数不马上返回错误,而将错误一一追加到一个错误列表里面。

// ToAggregate converts the ErrorList into an errors.Aggregate.
func (list ErrorList) ToAggregate() utilerrors.Aggregate {
    if len(list) == 0 {
        return nil
    }
    errs := make([]error, 0, len(list))
    errorMsgs := sets.NewString()
    for _, err := range list {
        msg := fmt.Sprintf("%v", err)
        if errorMsgs.Has(msg) {
            continue
        }
        errorMsgs.Insert(msg)
        errs = append(errs, err)
    }
    return utilerrors.NewAggregate(errs)
}

ToAggregate这个则是位于k8s.io/kubernetes/staging/src/k8s.io/apimachinery/pkg/util/validation/field/errors.go里封装的一个方法了。

在 Go 语言中,错误处理是一种非常重要的机制,可以帮助我们在程序运行时检测到错误,并采取相应的措施来解决问题。Go 语言中的错误类型是内置的 error 接口类型,其定义如下: ```go type error interface { Error() string } ``` 可以看到,这个接口只有一个方法 Error(),它返回一个字符串,表示错误的描述信息。因此,如果你想定义一个错误类型,只需要实现 error 接口的 Error() 方法即可。 在 Go 语言中,我们通常会使用函数返回值来传递错误信息。如果函数执行成功,通常会返回一个 nil 错误;如果函数执行失败,通常会返回一个非空的错误值,表示出错的具体信息。 例如,下面的代码演示了如何在打开一个文件时进行错误处理: ```go file, err := os.Open("test.txt") if err != nil { // 处理错误 fmt.Println("打开文件失败:", err.Error()) return } // 文件打开成功,进行操作 ``` 在这个代码中,我们使用 os.Open 函数打开一个文件。如果函数执行失败,会返回一个非空的错误值 err;否则,函数返回一个文件对象 file。我们可以使用 if err != nil 来检查 err 是否为空,如果不为空,表示函数执行失败,我们需要采取相应的措施来解决问题,例如打印错误信息并返回。如果 err 为空,则表示函数执行成功,我们可以继续操作文件对象。 除了使用 if err != nil 来检查错误之外,Go 语言还提供了一个更加简洁的语法,即使用 defer 和 panic 函数来处理错误。例如,下面的代码演示了如何在除数为零时触发 panic: ```go func divide(x, y int) int { defer func() { if err := recover(); err != nil { fmt.Println("出现了一个错误:", err) } }() if y == 0 { panic("除数不能为零") } return x / y } ``` 在这个代码中,我们使用 defer 和匿名函数来定义一个错误处理函数。在 divide 函数中,如果除数为零,我们会使用 panic 函数触发一个 panic,表示程序遇到了无法处理的错误。此时,defer 语句会立即执行匿名函数,该函数调用 recover 函数来捕获 panic,并打印错误信息。注意,在 defer 函数中使用 recover 函数可以避免程序崩溃,并返回一个错误信息。 总的来说,Go 语言提供了多种方式来处理错误,可以根据实际情况选择适合自己的方式。在实际开发中,我们通常会将错误信息记录到日志中,或者通过 HTTP 接口返回给客户端。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

王伯爵

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值