構造体を匿名クロージャとして埋め込むことは、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"}
ご覧のとおり、Data 構造体の Name フィールドは JSON 出力に含まれていません。 。これは、インターフェイスにはフィールドがないため、基になる構造体のフィールドはプロモートされないためです。
1 つの解決策は、リフレクションを使用して新しい構造体型を動的に生成することです。 。この構造体には、不明なタイプの匿名フィールドと追加データ用の追加フィールドが含まれます。
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"}
もう 1 つの方法は、未知の値をマーシャルし、それをマップにアンマーシャルし、追加のフィールドを追加して、マップをマーシャルし直すことです。
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 中国語 Web サイトの他の関連記事を参照してください。