
go 的标准 `json` 包默认按结构体字段名序列化,但可通过实现 `marshaljson()` 方法自定义序列化逻辑,将结构体字段(如 `name` 和 `value`)转为动态键值对,实现如 `{"country": "abc"}` 这类灵活 json 输出。
在 Go 中,JSON 序列化通常依赖结构体标签(如 json:"name")静态映射字段名,但当键名需由运行时数据(例如用户输入或配置)动态决定时,标准方式无法满足需求。此时,最简洁、符合 Go 惯用法的解决方案是为自定义类型实现 json.Marshaler 接口,即定义 MarshalJSON() ([]byte, error) 方法。
以下是一个完整、可直接使用的示例:
package main
import (
"encoding/json"
"fmt"
)
type xAxis struct {
Name string
Value string
}
// MarshalJSON 实现 json.Marshaler 接口
// 将{Name: "Country", Value: "abc"} 序列为 {"Country":"abc"}
func (a xAxis) MarshalJSON() ([]byte, error) {
// 构造仅含一个键值对的 map[string]interface{}
m := map[string]interface{}{a.Name: a.Value}
return json.Marshal(m)
}
func main() {
data := xAxis{
Name: "Country",
Value: "abc",
}
b, err := json.Marshal(data)
if err != nil {
panic(err)
}
fmt.Println(string(b)) // 输出:{"Country":"abc"}
}✅ 关键要点说明:
? 进阶提示:
如需统一处理多种类似结构(如 yAxis、filter),可抽象为泛型函数或封装为辅助类型(如 DynamicField{Key, Value}),避免重复实现。
总之,通过 MarshalJSON,你完全掌控了 Go 中 JSON 的输出形态——无需第三方库,不破坏类型安全,是构建灵活 API 响应的理想实践。