英文:
Reflecting The Fields of a Empty's Slices Underyling Type?
问题
我有以下的查询构建函数:
func CreateQuery(t interface{}, where string) {
var b bytes.Buffer
b.WriteString("SELECT ")
s := reflect.ValueOf(t).Elem()
typeOfT := s.Type()
for i := 0; i < s.NumField() - 1; i++ {
b.WriteString(fmt.Sprintf("%s, ", typeOfT.Field(i).Name))
}
// 最后一个字段没有逗号
b.WriteString(fmt.Sprintf("%s ", typeOfT.Field(s.NumField() - 1).Name))
b.WriteString(fmt.Sprintf("FROM %s ", typeOfT.Name()))
b.WriteString(where)
fmt.Println(b.String())
}
当以以下方式调用时,它可以正常工作:
var dst FooStruct
CreateQuery(&dst, "")
但是以下代码会引发一个"call of reflect.Value.NumField on slice Value"的错误:
var dst []FooStruct
CreateQuery(&dst, "")
我该如何修改函数以打印切片的底层结构类型的字段?看起来我需要反转reflect的SliceOf
函数。
英文:
I have the following query builder function:
func CreateQuery(t interface{}, where string) {
var b bytes.Buffer
b.WriteString("SELECT ")
s := reflect.ValueOf(t).Elem()
typeOfT := s.Type()
for i := 0; i < s.NumField() - 1; i++ {
b.WriteString(fmt.Sprintf("%s, ", typeOfT.Field(i).Name))
}
//Last one has no Comma
b.WriteString(fmt.Sprintf("%s ", typeOfT.Field(s.NumField() - 1).Name))
b.WriteString(fmt.Sprintf("FROM %s ", typeOfT.Name()))
b.WriteString(where)
fmt.Println(b.String())
}
There works fine when called as follows:
var dst FooStruct
CreateQuery(&dst, "")
But the following raises a "call of reflect.Value.NumField on slice Value" Panic:
var dst []FooStruct
CreateQuery(&dst, "")
How I can I make the function print the fields of a slice's underlying struct type? It seems like I want the inverse of reflect's SliceOf
function.
答案1
得分: 3
你只能在表示结构体的reflect.Type
上调用NumField
或Field
方法(即t.Kind() == reflect.Struct
)。
如果你有一个切片类型,你可以通过Elem
方法访问其中包含的类型,该方法返回另一个reflect.Type
。如果切片包含一个结构体,那么你可以在该类型上调用NumField
/Field
方法。
英文:
You can only call NumField
or Field
methods on a reflect.Type
representing a struct (i.e. t.Kind() == reflect.Struct
).
If you have a slice type, you can access the contained type via the Elem
method, which returns another reflect.Type
. If the slice contains a struct, then you can call NumField
/Field
on this type.
答案2
得分: 1
你可以遍历切片,在每个查询上调用CreateQuery
函数:
func CreateQueries(t interface{}, where string) {
v := reflect.ValueOf(t)
if v.Kind() == reflect.Ptr {
v = v.Elem()
}
if v.Kind() == reflect.Array || v.Kind() == reflect.Slice {
for i := 0; i < v.Len(); i++ {
CreateQuery(v.Index(i).Interface(), where)
}
}
}
使用reflect.Value.Index
可以单独访问每个字段,调用.Interface()
方法可以将值转换为interface{}
类型的表示,使其适用于放入CreateQuery
函数中(该函数期望一个interface{}
值)。
英文:
You can iterate over the slice, calling CreateQuery
for every query:
func CreateQueries(t interface{}, where string) {
v := reflect.ValueOf(t)
if v.Kind() == reflect.Ptr {
v = v.Elem()
}
if v.Kind() == reflect.Array || v.Kind() == reflect.Slice {
for i := 0; i < v.Len(); i++ {
CreateQuery(v.Index(i).Interface(), where)
}
}
}
Using reflect.Value.Index
you can access each field separately, calling .Interface()
on the
value yields the interface{}
type representation of that value, making it suitable to put
it in your CreateQuery
function (which expects a interface{}
value).
通过集体智慧和协作来改善编程学习和解决问题的方式。致力于成为全球开发者共同参与的知识库,让每个人都能够通过互相帮助和分享经验来进步。
评论