我如何得到struct的输出,按字段排序?
type T struct {
B int
A int
}
t := &T{B: 2, A: 1}
doSomething(t)
fmt.Println(t) // &{1 2} --> Sorted by fields
Run Code Online (Sandbox Code Playgroud)
A struct是有序的字段集合.该fmt包使用反射来获取值的字段和值struct,并按照定义它们的顺序生成输出.
因此,最简单的解决方案是声明您的类型,您已按字母顺序排列字段:
type T struct {
A int
B int
}
Run Code Online (Sandbox Code Playgroud)
如果无法修改字段的顺序(例如,内存布局很重要),则可以Stringer通过String()为结构类型指定方法来实现接口:
func (t T) String() string {
return fmt.Sprintf("{%d %d}", t.A, t.B)
}
Run Code Online (Sandbox Code Playgroud)
所述fmt包检查所传递的值实现Stringer,并且如果是的话,调用它的String()方法,以产生输出.
这个解决方案的缺点是它不灵活(例如,如果你添加一个新字段,你也必须更新String()方法),你也必须为struct你希望它工作的每个类型都这样做(你不能定义方法对于其他包中定义的类型).
完全灵活的解决方案可以使用反射.您可以获取字段的名称,按名称对它们进行排序,然后迭代排序的名称并获取字段值(按名称).
这个解决方案的优点是,这适用于任何一个struct,即使你从结构中添加或删除字段,它仍然可以不加修改地工作.它也适用于任何类型的字段,而不仅仅适用于int字段.
以下是如何操作的示例(在Go Playground上尝试):
func printFields(st interface{}) string {
t := reflect.TypeOf(st)
names := make([]string, t.NumField())
for i := range names {
names[i] = t.Field(i).Name
}
sort.Strings(names)
v := reflect.ValueOf(st)
buf := &bytes.Buffer{}
buf.WriteString("{")
for i, name := range names {
val := v.FieldByName(name)
if !val.CanInterface() {
continue
}
if i > 0 {
buf.WriteString(" ")
}
fmt.Fprintf(buf, "%v", val.Interface())
}
buf.WriteString("}")
return buf.String()
}
Run Code Online (Sandbox Code Playgroud)