Golang,按字母顺序对结构字段进行排序

tom*_*ian 2 sorting struct go

我如何得到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)

icz*_*cza 5

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)