
当已有合法 JSON 格式的字符串(如从外部获取或动态拼接)时,不应再次 json.Marshal 或 json.Encode,否则会导致双重编码;直接以 UTF-8 字节形式写入 http.ResponseWriter 即可确保响应内容为原始 JSON。
当已有合法 json 格式的字符串(如从外部获取或动态拼接)时,不应再次 `json.marshal` 或 `json.encode`,否则会导致双重编码;直接以 utf-8 字节形式写入 `http.responsewriter` 即可确保响应内容为原始 json。
在 Go 的 HTTP 服务开发中,一个常见误区是:误将已格式化好的 JSON 字符串当作普通 Go 值进行序列化。例如,你手头已有一个符合 JSON 语法的字符串 "{\"key1\":{\"key2\":\"value1\",\"key3\":\"value2\"}}",此时若调用 json.NewEncoder(w).Encode(str),Go 会把该字符串作为一个 Go string 类型值再次转义并封装成 JSON —— 最终响应体变成:
"{\"key1\":{\"key2\":\"value1\",\"key3\":\"value2\"}}"即外层多了一对双引号和转义斜杠,不再是有效的嵌套 JSON 对象,而是一个 JSON 字符串字面量。
✅ 正确做法是:跳过序列化环节,直接写入原始字节。因为该字符串本身已是合法、完整的 JSON 文本,只需确保:
- 内容以 UTF-8 编码;
- 响应头 Content-Type 设置为 application/json; charset=utf-8;
- 使用 w.Write([]byte(str)) 或更安全的 fmt.Fprint(w, str)。
示例代码如下:
func handler(w http.ResponseWriter, r *http.Request) {
str := "{\"key1\":{\"key2\":\"value1\",\"key3\":\"value2\"}}"
// ✅ 正确:设置 Content-Type 并直接写入字节
w.Header().Set("Content-Type", "application/json; charset=utf-8")
if _, err := w.Write([]byte(str)); err != nil {
http.Error(w, "Failed to write response", http.StatusInternalServerError)
return
}
}⚠️ 注意事项:
- 务必验证输入字符串是否为合法 JSON:若 str 来自不可信来源(如用户输入、第三方 API),需先用 json.Unmarshal 尝试解析,避免返回无效 JSON 导致前端解析失败;
- 不要混用 json.Encode 和原始写入:json.NewEncoder(w).Encode(...) 适用于 Go 值(如 map[string]interface{}、结构体等),而非 JSON 字符串;
- 避免手动拼接 JSON 字符串:长期来看,应优先使用 Go 原生结构体 + json.Marshal 构建响应,既类型安全又免于转义错误。
总结:Go 的 json 包设计面向「Go 值 → JSON 字节」的转换;若你已持有 JSON 字节(表现为 string),就应绕过序列化,直写响应体——这是高效、准确且符合 HTTP 语义的最佳实践。










