将结构体嵌入为匿名闭包是将额外字段添加到 JSON 的常见方法输出。然而,这种技术仅限于结构类型已知的情况。
在处理任意结构或接口时,传统的嵌入方法会失败。考虑以下代码:
type example interface{} type Data struct { Name string } type Extra struct { Text string } func printInterface(val interface{}) { example1 := struct { example Extra string }{ example: val, Extra: "text", } json.NewEncoder(os.Stdout).Encode(example1) } func main() { d := Data{Name:"name"} printInterface(&d) }
此代码生成以下 JSON 输出:
{"example":{"Name":"name"},"Extra":"text"}
如您所见,数据结构的 Name 字段不包含在 JSON 输出中。这是因为接口没有字段,因此底层结构体的字段不会被提升。
一种解决方案是使用反射动态生成新的结构体类型。此结构将包含未知类型的匿名字段和用于额外数据的附加字段。
func printInterface(val interface{}) { t2 := reflect.StructOf([]reflect.StructField{ reflect.StructField{ Name: "X", Anonymous: true, Type: reflect.TypeOf(val), }, reflect.StructField{ Name: "Extra", Type: reflect.TypeOf(""), }, }) v2 := reflect.New(t2).Elem() v2.Field(0).Set(reflect.ValueOf(val)) v2.FieldByName("Extra").SetString("text") json.NewEncoder(os.Stdout).Encode(v2.Interface()) }
此代码生成所需的 JSON 输出:
{"Name":"name","Extra":"text"}
另一种选择是编组未知值,将其解组到映射中,添加额外的字段,然后将映射重新编组到JSON.
func printInterface(val interface{}) error { data, err := json.Marshal(val) if err != nil { return err } v2 := map[string]interface{}{} if err := json.Unmarshal(data, &v2); err != nil { return err } v2["Extra"] = "text" return json.NewEncoder(os.Stdout).Encode(v2) }
此解决方案更简单,但由于双重编组过程,效率可能较低。
以上是如何在 Go 中向未知结构的 JSON 输出添加任意字段?的详细内容。更多信息请关注PHP中文网其他相关文章!