Go 中是否有通用的辅助方法可以根据reflect.Kind
将string
转换为正确的值?
还是我需要自己实现各种切换?
我有一个像"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
,你可以利用例如已经包含这个switch
的json
包来解码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)