使用反射附加到结构中的切片字段

Son*_*ton 7 reflection go

我有一个struct看起来像这样:

type guitaristT struct {
    Surname  string   `required=true`
    Year     int64    `required=false`
    American bool     // example of missing tag
    Rating   float32  `required=true`
    Styles   []string `required=true,minsize=1`
}
Run Code Online (Sandbox Code Playgroud)

我有一个如下所示的环境变量,我使用反射来填充基于键的结构。

jimiEnvvar :="surname=Hendrix|year=1942|american=true|rating=9.99
  |styles=blues|styles=rock|styles=psychedelic"
Run Code Online (Sandbox Code Playgroud)

我可以string, int64, bool and float32使用反射设置字段,但我坚持如何附加到slice字段Styles。例如,基于上述,jimiEnvvar我希望该字段jimi.Styles具有 values ["blues","rock", "psychedelic"]

我有以下(简化)代码:

result := guitaristT{}
// result.Styles = make([]string, 10) // XXX causes 10 empty strings at start
result.Styles = make([]string, 0)     // EDIT: Alessandro's solution
...
v := reflect.ValueOf(&result).Elem()
...
field := v.FieldByName(key) // eg key = "styles"
...
switch field.Kind() {

case reflect.Slice:
     // this is where I get stuck
     //
     // reflect.Append() has signature:
     //   func Append(s Value, x ...Value) Value

     // so I convert my value to a reflect.Value
     stringValue := reflect.ValueOf(value) // eg value = "blues"

     // field = reflect.Append(field, stringValue)  // XXX doesn't append
     field.Set(reflect.Append(field, stringValue))  // EDIT: Alessandro's solution
Run Code Online (Sandbox Code Playgroud)

编辑:

第二部分(我解决了)是在结构中填充地图。例如:

type guitaristT struct {
    ...
    Styles   []string `required=true,minsize=1`
    Cities   map[string]int
}
Run Code Online (Sandbox Code Playgroud)

jimiEnvvar 的样子:

jimiEnvvar += "|cities=New York^17|cities=Los Angeles^14"
Run Code Online (Sandbox Code Playgroud)

我是这样写的:

    case reflect.Map:
        fmt.Println("keyAsString", keyAsString, "is Map, has value:", valueAsString)
        mapKV := strings.Split(valueAsString, "^")
        if len(mapKV) != 2 {
            log.Fatalln("malformed map key/value:", mapKV)
        }
        mapK := mapKV[0]
        mapV := mapKV[1]
        thisMap := fieldAsValue.Interface().(map[string]int)
        thisMap[mapK] = atoi(mapV)
        thisMapAsValue := reflect.ValueOf(thisMap)
        fieldAsValue.Set(thisMapAsValue)
Run Code Online (Sandbox Code Playgroud)
The final result was:

main.guitaristT{
    Surname:  "Hendrix",
    Year:     1942,
    American: true,
    Rating:   9.989999771118164,
    Styles:   {"blues", "rock", "psychedelic"},
    Cities:   {"London":11, "Bay Area":9, "New York":17, "Los Angeles":14},
}
Run Code Online (Sandbox Code Playgroud)

如果您有兴趣,完整代码位于https://github.com/soniah/reflect/blob/master/structs.go。代码只是我写的一些笔记/练习。


编辑2:

“Go in Practice”(Butcher 和 Farina)的第 11 章详细解释了反射、结构和标签。

Ale*_*ini 3

你离得并不遥远。只需替换为

field.Set(reflect.Append(field, stringValue)) 
Run Code Online (Sandbox Code Playgroud)

你就完成了。另外,请确保使用初始化切片

result.Styles = make([]string, 0)
Run Code Online (Sandbox Code Playgroud)

否则数组顶部将出现 10 个空白字符串Styles

希望这对您有所帮助并祝您的项目顺利。