0

0

Golang JSON序列化:理解字段可见性与struct标签的应用

聖光之護

聖光之護

发布时间:2025-11-24 16:27:59

|

999人浏览过

|

来源于php中文网

原创

Golang JSON序列化:理解字段可见性与struct标签的应用

本文深入探讨了go语言中结构体字段在json序列化时遇到的常见问题:当结构体字段以小写字母开头时,`json.marshal`为何无法正确生成json输出。文章详细解释了go语言的字段可见性规则(导出与未导出),阐明了`json.marshal`仅能访问导出字段的原理。此外,教程还提供了两种解决方案:将字段名首字母大写以导出字段,以及使用`json` struct标签来自定义json输出字段名,即使go字段名已导出,也能实现灵活的json映射。

理解Go语言的字段可见性与JSON序列化

在Go语言中,进行JSON序列化时,开发者可能会遇到一个常见且令人困惑的现象:当结构体(struct)的字段名以小写字母开头时,json.Marshal函数生成的JSON字符串会是一个空的JSON对象{}。而将字段名首字母改为大写后,序列化则能正常进行。这并非Go语言的bug,而是其核心设计理念——可见性规则(Visibility Rules)——在起作用。

问题现象示例

考虑以下Go结构体定义:

type Machine struct {
  m_ip     string
  m_type   string
  m_serial string
}

当我们尝试将其序列化为JSON时:

import (
    "encoding/json"
    "fmt"
)

func main() {
    m := &Machine{m_ip: "192.168.1.1", m_type: "Server", m_serial: "ABC123XYZ"}
    m_json, err := json.Marshal(m)
    if err != nil {
        fmt.Println("Error marshalling:", err)
        return
    }
    fmt.Println(string(m_json)) // 输出: {}
}

上述代码的输出将是{},这表明结构体中的字段并未被正确序列化。

立即学习go语言免费学习笔记(深入)”;

Go语言的可见性规则

Go语言使用首字母的大小写来决定一个标识符(变量、函数、结构体字段等)在其包内外的可见性:

  • 导出(Exported)标识符: 如果标识符的首字母是大写,则它是一个导出标识符。这意味着它可以在当前包外部被访问和使用。
  • 未导出(Unexported)标识符: 如果标识符的首字母是小写,则它是一个未导出标识符。这意味着它只能在当前包内部被访问和使用。

json.Marshal函数是encoding/json包的一部分。当它尝试序列化一个结构体时,它需要访问结构体中的字段。根据Go的可见性规则,json.Marshal只能“看到”并访问那些被导出的字段。对于未导出的字段(即首字母小写的字段),json.Marshal无法访问,因此在序列化时会被忽略,导致它们不会出现在最终的JSON输出中。

在上面的示例中,m_ip、m_type和m_serial都是以小写字母开头的,它们是未导出的字段,因此json.Marshal无法对其进行序列化。

解决方案一:导出结构体字段

最直接的解决方案是将结构体中需要序列化的字段首字母改为大写,使其成为导出字段。

type Machine struct {
  MachIp     string
  MachType   string
  MachSerial string
}

现在,当我们再次尝试序列化:

Memories.ai
Memories.ai

专注于视频解析的AI视觉记忆模型

下载
import (
    "encoding/json"
    "fmt"
)

func main() {
    m := &Machine{MachIp: "192.168.1.1", MachType: "Server", MachSerial: "ABC123XYZ"}
    m_json, err := json.Marshal(m)
    if err != nil {
        fmt.Println("Error marshalling:", err)
        return
    }
    fmt.Println(string(m_json))
}

输出将是:

{"MachIp":"192.168.1.1","MachType":"Server","MachSerial":"ABC123XYZ"}

这证明了将字段导出后,json.Marshal能够成功访问并序列化它们。

解决方案二:使用JSON Struct Tag(推荐)

虽然将字段首字母大写可以解决序列化问题,但在某些情况下,我们可能希望JSON输出的字段名保持小写或具有特定的命名风格(例如,snake_case)。这时,Go语言的struct tag就派上用场了。

通过在结构体字段声明后添加反引号(`)包裹的tag,我们可以为json.Marshal提供额外的指令,告诉它在序列化时如何处理该字段。对于JSON序列化,我们使用json:"field_name"的格式。

type Machine struct {
    MachIp     string `json:"m_ip"`
    MachType   string `json:"m_type"`
    MachSerial string `json:"m_serial"`
}

在这个例子中:

  • MachIp、MachType、MachSerial仍然是导出字段(首字母大写),因此json.Marshal可以访问它们。
  • json:"m_ip"这样的tag告诉json.Marshal,在生成JSON时,将MachIp字段映射为m_ip。

现在,再次运行序列化代码:

import (
    "encoding/json"
    "fmt"
)

func main() {
    m := &Machine{MachIp: "192.168.1.1", MachType: "Server", MachSerial: "ABC123XYZ"}
    m_json, err := json.Marshal(m)
    if err != nil {
        fmt.Println("Error marshalling:", err)
        return
    }
    fmt.Println(string(m_json))
}

输出将是:

{"m_ip":"192.168.1.1","m_type":"Server","m_serial":"ABC123XYZ"}

通过使用struct tag,我们既遵循了Go语言的可见性规则(字段为导出),又实现了JSON输出字段名的自定义,这在与外部API或前端进行数据交互时非常有用。

总结与注意事项

  • 核心原则: json.Marshal只能序列化结构体中导出(首字母大写)的字段。
  • 解决方案:
    1. 将结构体字段的首字母改为大写,使其成为导出字段。
    2. 使用json:"field_name" struct tag来自定义JSON输出字段名,同时保持Go结构体字段为导出状态。
  • 最佳实践: 在Go语言中,推荐始终将需要序列化或在包外访问的结构体字段定义为导出字段(首字母大写)。如果需要自定义JSON输出的字段名,应优先使用struct tag,这提供了更大的灵活性和更好的代码可读性
  • 忽略字段: 如果希望某个导出字段不被序列化,可以使用json:"-" tag。
  • 空值处理: 使用json:"field_name,omitempty" tag可以在字段值为其零值(例如,字符串为空,整数为0,切片为nil)时,不将其包含在JSON输出中。

理解Go语言的可见性规则及其在JSON序列化中的应用,是编写健壮和符合Go惯例代码的关键。通过合理利用struct tag,可以更灵活地控制JSON的输入和输出格式。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

180

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

228

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

340

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

209

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

393

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

197

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

191

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

253

2025.06.17

菜鸟裹裹入口以及教程汇总
菜鸟裹裹入口以及教程汇总

本专题整合了菜鸟裹裹入口地址及教程分享,阅读专题下面的文章了解更多详细内容。

0

2026.01.22

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.4万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号