使用Golang中的类型断言检测超出范围错误的值

时间:2015-03-11 02:45:59

标签: go type-assertion strconv

给出以下代码:

iv, err := strconv.ParseInt("18446744073709551448", 10, 64)
fmt.Println(iv)
fmt.Printf("%#v\n", err)
fmt.Printf("%v\n", err)

//Output:
9223372036854775807
&strconv.NumError{Func:"ParseInt", Num:"18446744073709551448", Err:(*errors.errorString)(0x1040a040)}
strconv.ParseInt: parsing "18446744073709551448": value out of range

如何检测到函数因int64超出范围而失败? strconv.ParseInt函数返回错误类型,但在这种情况下,它实际上是strconv.NumError类型,由%#v表示。 Error Handling and Go文章提到您可以使用类型断言来检查特定类型的错误,但它没有给出任何示例。我应该使用什么表达式来完成此代码:

if expression {
    uv, err := strconv.ParseUint("18446744073709551448", 10, 64)
}

2 个答案:

答案 0 :(得分:10)

我们有,

  

Package strconv

     

var ErrRange = errors.New("value out of range")

     

ErrRange表示某个值超出了目标类型的范围。

type NumError struct {
        Func string // the failing function (ParseBool, ParseInt, ParseUint, ParseFloat)
        Num  string // the input
        Err  error  // the reason the conversion failed (ErrRange, ErrSyntax)
}
     

NumError记录转换失败。

func (e *NumError) Error() string

例如,

package main

import (
    "fmt"
    "strconv"
)

func main() {
    iv, err := strconv.ParseInt("18446744073709551448", 10, 64)
    if err != nil {
        if numError, ok := err.(*strconv.NumError); ok {
            if numError.Err == strconv.ErrRange {
                fmt.Println("Detected", numError.Num, "as a", strconv.ErrRange)
                return
            }
        }
        fmt.Println(err)
        return
    }
    fmt.Println(iv)
}

输出:

Detected 18446744073709551448 as a value out of range

答案 1 :(得分:-1)

strconv.ParseInt返回的错误仅在编译时被称为某种实现Error接口的类型。类型断言允许你坚持它是一个strconv.NumError并直接检查它的字段,但如果结果是错误则冒着运行时恐慌的风险:

if err.(*strconv.NumError).Err.Error() == "value out of range" {
    uv, err := strconv.ParseUint("18446744073709551448", 10, 64)
}

更灵活的解决方案(但可能太松散 - 用于您的目的)将在err.Error()方法上执行子字符串匹配:

if strings.Contains(err.Error(), "value out of range") {
    uv, err := strconv.ParseUint("18446744073709551448", 10, 64)
}