如何在 Go 中将嵌套结构体正确序列化为 JSON
发布时间 - 2026-01-04 00:00:00 点击率:次本文详解 go 语言中嵌套结构体的 json 序列化方法,重点解决匿名嵌套结构体字面量语法错误问题,并提供可维护、符合 api 规范的命名类型 + json 标签最佳实践。
在 Go 中将嵌套结构体(尤其是含匿名字段的结构体)正确 json.Marshal 成期望格式(如 {"genre": {"country": "taylor swift", "rock": "aimee"}}),关键在于理解 复合字面量(composite literal)的语法要求 和 JSON 字段映射的控制方式。
❌ 错误写法分析
原始代码报错 Missing type in composite literal,原因在于:
type Music struct {
Genre struct {
Country string
Rock string
}
}
resp := Music{
Genre: { // ⚠️ 缺失类型信息!Go 无法推断此处应构造哪个匿名 struct
Country: "Taylor Swift",
Rock: "Aimee",
},
}Go 要求每个复合字面量必须显式声明其类型。即使结构体字段是匿名的,初始化时仍需重复该匿名类型的完整定义。
✅ 正确方案一:显式匿名类型字面量(仅作演示,不推荐生产使用)
resp := Music{
Genre: struct {
Country string
Rock string
}{ // ✅ 显式写出类型 + 花括号初始化
Country: "Taylor Swift",
Rock: "Aimee",
},
}虽然语法合法,但重复冗长、难以复用、不可导出,且无法添加 JSON 字段标签(如 json:"country"),导致输出键名默认为大写首字母("Country" → "Country"),不符合 RESTful API 常见的小驼峰或 snake_case 规范。
✅ 推荐方案:定义命名结构体 + JSON 标签(生产级写法)
type Genre struct {
Country string `json:"country"`
Rock string `json:"rock"`
}
type Music struct {
Genre Genre `json:"genre"`
}
resp := Music{
Genre: Genre{ // ✅ 类型清晰、可复用、支持标签
Country: "Taylor Swift",
Rock: "Aimee",
},
}
data, err := json.Marshal(resp)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
w.Header().Set("Content-Type", "application/json")
w.Write(data)
// 输出:{"genre":{"country":"Taylor Swift","rock":"Aimee"}}✅ 优势总结:语义清晰:Genre 类型明确表达业务含义;可复用性强:可在其他结构体(如 Album、ArtistProfile)中复用;精准控制 JSON 键名:通过 `json:"country"` 实现小写字母键;支持导出与文档化:便于生成 Swagger 文档或团队协作;利于测试与扩展:可为 Genre 添加方法(如 Validate()、String())。
? 补充说明:字段可见性与 JSON 序列化规则
- 只有首字母大写的导出字段才会被 json.Marshal 序列化;
- 小写字母开头的字段(如 country string)是未导出的,会被忽略;
- 若需强制忽略某字段,使用 `json:"-"`;若需零值也输出,加 ,omitempty(如 `json:"country,omitempty"`)。
? 完整可运行示例
package main
import (
"encoding/json"
"fmt"
"log"
)
type Genre struct {
Country string `json:"country"`
Rock string `json:"rock"`
}
type Music struct {
Genre Genre `json:"genre"`
}
func main() {
resp := Music{
Genre: Genre{
Country: "Taylor Swift",
Rock: "Aimee",
},
}
dat
a, err := json.Marshal(resp)
if err != nil {
log.Fatal(err)
}
fmt.Println(string(data)) // {"genre":{"country":"Taylor Swift","rock":"Aimee"}}
}遵循此模式,不仅能彻底规避 Missing type in composite literal 错误,更能构建出结构清晰、可维护、符合工业标准的 Go JSON API 响应体系。
# js
# json
# go
# app
# ai
# restful api
# swift
# restful
# String
# 结构体
# 复用
# 序列化
# 首字母
# 若需
# 文档
# 尤其是
# 键名
# 才会
# 可在
# 不符合
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
微信小程序 HTTPS报错整理常见问题及解决方案
如何在Ubuntu系统下快速搭建WordPress个人网站?
微信小程序 五星评分(包括半颗星评分)实例代码
网站页面设计需要考虑到这些问题
如何基于云服务器快速搭建个人网站?
如何在Tomcat中配置并部署网站项目?
Laravel N+1查询问题如何解决_Eloquent预加载(Eager Loading)优化数据库查询
零服务器AI建站解决方案:快速部署与云端平台低成本实践
C#如何调用原生C++ COM对象详解
JavaScript 输出显示内容(document.write、alert、innerHTML、console.log)
bootstrap日历插件datetimepicker使用方法
如何用已有域名快速搭建网站?
如何在Windows虚拟主机上快速搭建网站?
Linux系统命令中tree命令详解
php增删改查怎么学_零基础入门php数据库操作必知基础【教程】
如何在阿里云香港服务器快速搭建网站?
网站制作公司哪里好做,成都网站制作公司哪家做得比较好,更正规?
Laravel如何使用模型观察者?(Observer代码示例)
Laravel如何使用软删除(Soft Deletes)功能_Eloquent软删除与数据恢复方法
如何用AWS免费套餐快速搭建高效网站?
如何快速查询网址的建站时间与历史轨迹?
郑州企业网站制作公司,郑州招聘网站有哪些?
Laravel如何使用Collections进行数据处理?(实用方法示例)
Java类加载基本过程详细介绍
如何快速生成专业多端适配建站电话?
Microsoft Edge如何解决网页加载问题 Edge浏览器加载问题修复
PythonWeb开发入门教程_Flask快速构建Web应用
如何在 Python 中将列表项按字母顺序编号(a.、b.、c. …)
Win11怎么恢复误删照片_Win11数据恢复工具使用【推荐】
宙斯浏览器视频悬浮窗怎么开启 边看视频边操作其他应用教程
如何在景安云服务器上绑定域名并配置虚拟主机?
大连企业网站制作公司,大连2025企业社保缴费网上缴费流程?
Mybatis 中的insertOrUpdate操作
Laravel怎么进行数据库回滚_Laravel Migration数据库版本控制与回滚操作
如何快速查询网站的真实建站时间?
LinuxCD持续部署教程_自动发布与回滚机制
Python面向对象测试方法_mock解析【教程】
Win11怎么更改系统语言为中文_Windows11安装语言包并设为显示语言
如何在腾讯云服务器上快速搭建个人网站?
简单实现Android文件上传
Win11应用商店下载慢怎么办 Win11更改DNS提速下载【修复】
香港服务器网站推广:SEO优化与外贸独立站搭建策略
如何在云服务器上快速搭建个人网站?
Laravel中Service Container是做什么的_Laravel服务容器与依赖注入核心概念解析
Linux系统运维自动化项目教程_Ansible批量管理实战
企业网站制作这些问题要关注
如何在IIS中新建站点并配置端口与物理路径?
iOS验证手机号的正则表达式
Laravel如何生成URL和重定向?(路由助手函数)
如何快速搭建高效WAP手机网站吸引移动用户?


a, err := json.Marshal(resp)
if err != nil {
log.Fatal(err)
}
fmt.Println(string(data)) // {"genre":{"country":"Taylor Swift","rock":"Aimee"}}
}