所以我找到了一些代码来帮助我开始使用Go(golang)中的反射,但是我在获取基础值时遇到了麻烦,因此我基本上可以从结构中创建map[string]string
并且它的领域。
最后,我想将结果变成map[string]interface{}
,但这个问题阻止了我。
我现在的代码:
package main
import (
"fmt"
"reflect"
)
type Foo struct {
FirstName string `tag_name:"tag 1"`
LastName string `tag_name:"tag 2"`
Age int `tag_name:"tag 3"`
}
func inspect(f interface{}) map[string]string {
m := make(map[string]string)
val := reflect.ValueOf(f).Elem()
for i := 0; i < val.NumField(); i++ {
valueField := val.Field(i)
typeField := val.Type().Field(i)
f := valueField.Interface()
val := reflect.ValueOf(f)
m[typeField.Name] = val.String()
}
return m
}
func dump(m map[string]string) {
for k, v := range m {
fmt.Printf("%s : %s\n", k, v)
}
}
func main() {
f := &Foo{
FirstName: "Drew",
LastName: "Olson",
Age: 30,
}
a := inspect(f)
dump(a)
}
运行代码的输出:
FirstName : Drew
LastName : Olson
Age : <int Value>
据我所知,FirstName和LastName的输出是实际的reflect.Value对象但是对于字符串,String上的String()方法只输出底层的String。我想要获取int并将其更改为字符串,但是从relfect包文档中我并没有立即看到它是如何完成的。
Soo ....如何从golang中的reflect.Value中获取基础值?
答案 0 :(得分:14)
如何解析值的一个很好的例子是fmt
包。请参阅this code。
使用上述代码来匹配您的问题将如下所示:
switch val.Kind() {
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
m[typeField.Name] = strconv.FormatInt(val.Int(), 10)
case reflect.String:
m[typeField.Name] = val.String()
// etc...
}
基本上您需要检查所有available Kinds。
答案 1 :(得分:3)
看起来你走在正确的轨道上。我在代码中看到的问题是它对值进行了假设,这意味着何时调用Elem()
和多少次(以解析指针)。要了解这一点,您需要查看reflect.Kind
。值是reflect.Ptr
吗?然后使用Elem()
。
获得val.Interface()
/ val.String()
/ val.Int()
的值后,您可以根据需要转换值。你使用的将取决于reflect.Kind
。要将int
转换为string
,您需要使用strconv
包。
encoding/json
和encoding/xml
包已经完成了这项工作。源代码提供了一些很好的例子。例如,请查看encoding/xml/read.go中的copyValue
和encoding/xml/marshal.go中的marshalSimple
。
答案 2 :(得分:2)
Go 1.5应该更容易(2015年8月)
review 8731旁边的commit 049b89d和Rob Pike (robpike
):
fmt
:特别对待reflect.Value
- 作为其持有的值
这将允许您打印Reflect.Value()
参数的实际值:
当
reflect.Value
传递给Printf
(等)时,fmt
会调用String
方法,但不会透露其内容。
要获取内容,可以调用Value.Interface()
,但这是非法的 如果Value
未导出或以其他方式被禁止。此CL通过对
fmt
包进行微不足道的更改来改善这种情况:当我们看到reflect.Value
作为参数时,我们会将其视为我们对待reflect.Value
包裹。
这意味着我们始终打印Value
的内容,就像 是Printf
的参数一样。这可以说是一个突破性的变化,但我认为这是一个真正的改进,并没有比我们对这个包的格式化输出所做的许多其他调整更大的突破。
答案 3 :(得分:1)
另一个简单的解决方案可以是,
flavorName = fmt.Sprintf("%v",strct)
“ fmt.Sprintf()”将返回可以存储在变量中的值。