直接用结构体指针和预编译编码器、复用 json.Encoder 与缓冲区、绕过反射、避免中间表示、实现 MarshalJSON 接口,可显著提升 JSON 生成效率。
直接用结构体指针和预编译的编码器,跳过反射、复用缓冲区,就能显著提升 JSON 生成效率。关键不是“怎么序列化”,而是“怎么不重复做没必要的事”。
每次调用 json.Marshal 都会分配新缓冲区、走完整反射路径;而 json.Encoder 可绑定到 bytes.Buffer 或网络连接,支持多次写入且内部缓存类型信息。
encoder := json.NewEncoder(buf),后续用 encoder.Encode(v) 写入多个值w(http.ResponseWriter 实现了 io.Writer),避免中间拷贝buf.Reset() 可循环使用同一缓冲区,减少 GC 压力标准库 json 包在首次处理某结构体时会构建字段映射表,之后复用;但若结构体类型多、冷启动频繁(如微服务启停),仍会触发重复计算。可手动缓存或使用代码生成。
json:"name")固定,Go 运行时会自动缓存其编码器map[string]interface{} —— 它们每次都会触发完整反射MarshalJSON 方法,零反射;后者用 unsafe + 类型系统优化,比标准库快 2–5 倍不要先 Marshal 成 []byte,再转 string 或拼接;更不要把 JSON 当字符串解析再改字段——这等于做两次序列化+一次反序列化。
map[string]any 而非 map[string]interface{}(Go 1.18+ 更高效),且尽量控制键数量json.RawMessage 延迟解析原始字节,仅对变动字段重新编码
encoder.Encode 连续写入,配合 bytes.Buffer 的 Grow 预分配容量,减少扩容拷贝对高频结构体,实现 json.Marshaler 接口,手写扁平、确定的编码逻辑,彻底跳过反射和 tag 解析。
func (t MyTime) MarshalJSON() ([]byte, error) { return []byte(`"` + t.Time.Format("2006-01-02T15:04:05Z") + `"`), nil }MarshalJSON 中显式判断,不依赖 omitempty 的反射检查