如何将字符串值转换为正确的反射.善良在去



Go 中是否有通用的辅助方法可以根据reflect.Kindstring转换为正确的值?

还是我需要自己实现各种切换?

我有一个像"143"这样的值作为字符串和一个反射。类型为"UInt16"的值,喜欢转换该字符串值并将其设置为我的结构的 UInt16 值。

我当前的代码如下所示:

func setValueFromString(v reflect.Value, strVal string) error {
    switch v.Kind() {
    case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
        val, err := strconv.ParseInt(strVal, 0, 64)
        if err != nil {
            return err
        }
        if v.OverflowInt(val) {
            return errors.New("Int value too big: " + strVal)
        }
        v.SetInt(val)
    case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
        val, err := strconv.ParseUint(strVal, 0, 64)
        if err != nil {
            return err
        }
        if v.OverflowUint(val) {
            return errors.New("UInt value too big: " + strVal)
        }
        v.SetUint(val)
    case reflect.Float32:
        val, err := strconv.ParseFloat(strVal, 32)
        if err != nil {
            return err
        }
        v.SetFloat(val)
    case reflect.Float64:
        val, err := strconv.ParseFloat(strVal, 64)
        if err != nil {
            return err
        }
        v.SetFloat(val)
    case reflect.String:
        v.SetString(strVal)
    case reflect.Bool:
        val, err := strconv.ParseBool(strVal)
        if err != nil {
            return err
        }
        v.SetBool(val)
    default:
        return errors.New("Unsupported kind: " + v.Kind().String())
    }
    return nil
}

这已经有效了,但我想知道这是否已经在其他地方实现。

编辑:原始问题的答案("如何从其string表示中获取reflect.Kind"(在最后。对编辑问题的回答如下:


您正在做的是最快和"最安全"的。如果你不想麻烦那么大的switch,你可以利用例如已经包含这个switchjson包来解码JSON字符串中的值(在encoding/json/decode.go,未导出的函数literalStore()(。

您的解码函数可能如下所示:

func Set(v interface{}, s string) error {
    return json.Unmarshal([]byte(s), v)
}

一个简单的电话json.Unmarshal().使用/测试它:

{
    var v int
    err := Set(&v, "1")
    fmt.Println(v, err)
}
{
    var v int
    err := Set(&v, "d")
    fmt.Println(v, err)
}
{
    var v uint32
    err := Set(&v, "3")
    fmt.Println(v, err)
}
{
    var v bool
    err := Set(&v, "true")
    fmt.Println(v, err)
}
{
    var v float32
    err := Set(&v, `5.1`)
    fmt.Println(v, err)
}
{
    var v string
    err := Set(&v, strconv.Quote("abc"))
    fmt.Println(v, err)
}

需要注意的一件事:当您要传递string时,必须引用,例如用strconv.Quote() 。输出(在Go Playground上尝试(:

1 <nil>
0 invalid character 'd' looking for beginning of value
3 <nil>
true <nil>
5.1 <nil>
abc <nil>

如果你不想要求带引号的字符串(这只会使事情复杂化(,你可以把它构建到 Set() 函数中:

func Set(v interface{}, s string) error {
    if t := reflect.TypeOf(v); t.Kind() == reflect.Ptr &&
        t.Elem().Kind() == reflect.String {
        s = strconv.Quote(s)
    }
    return json.Unmarshal([]byte(s), v)
}

然后你可以用一个string变量的地址和一个不加引号的string值来调用它:

var v string
err := Set(&v, "abc")
fmt.Println(v, err)

在Go Playground上尝试这种变体。


对原始问题的回答:如何从其string表示中获取reflect.Kind

reflect.Kind声明 :

type Kind uint

reflect.Kind s 的不同值是常量:

const (
    Invalid Kind = iota
    Bool
    Int
    Int8
    // ...
    Struct
    UnsafePointer
)

reflect包仅为reflect.Kind()类型提供一种方法:

func (k Kind) String() string

因此,就目前而言,您无法从其string表示中获取reflect.Kind(使用 Kind.String() 方法只能实现相反的方向(。但是提供此功能并不难。

我们要做的是构建一个由各种map

var strKindMap = map[string]reflect.Kind{}

我们像这样初始化它:

func init() {
    for k := reflect.Invalid; k <= reflect.UnsafePointer; k++ {
        strKindMap[k.String()] = k
    }
}

这是可能且正确的,因为常量是使用 iota 初始化的,计算结果为连续的非类型化整数常量,第一个值reflect.Invalid,最后一个值reflect.UnsafePointer

现在,您只需索引此地图即可从其string表示中获取reflect.Kind。一个帮助程序函数,它这样做:

func strToKind(s string) reflect.Kind {
    k, ok := strKindMap[s]
    if !ok {
        return reflect.Invalid
    }
    return k
}

我们完成了。测试/使用它:

fmt.Printf("All: %#vn", strKindMap)
for _, v := range []string{"Hey", "uint8", "ptr", "func", "chan", "interface"} {
    fmt.Printf("String: %q, Kind: %v (%#v)n", v, strToKind(v), strToKind(v))
}

输出(在Go Playground上尝试(:

All: map[string]reflect.Kind{"int64":0x6, "uint8":0x8, "uint64":0xb, "slice":0x17, "uintptr":0xc, "int8":0x3, "array":0x11, "interface":0x14, "unsafe.Pointer":0x1a, "complex64":0xf, "complex128":0x10, "int":0x2, "uint":0x7, "int16":0x4, "uint16":0x9, "map":0x15, "bool":0x1, "int32":0x5, "ptr":0x16, "string":0x18, "func":0x13, "struct":0x19, "invalid":0x0, "uint32":0xa, "float32":0xd, "float64":0xe, "chan":0x12}
String: "Hey", Kind: invalid (0x0)
String: "uint8", Kind: uint8 (0x8)
String: "ptr", Kind: ptr (0x16)
String: "func", Kind: func (0x13)
String: "chan", Kind: chan (0x12)
String: "interface", Kind: interface (0x14)

相关内容

  • 没有找到相关文章

最新更新