0

0

解决 Go 语言 json.Marshal 导出空对象问题:字段可见性详解

DDD

DDD

发布时间:2025-11-25 15:39:07

|

354人浏览过

|

来源于php中文网

原创

解决 Go 语言 json.Marshal 导出空对象问题:字段可见性详解

在使用 go 语言的 `encoding/json` 包进行 json 序列化时,如果自定义结构体(struct)的字段以小写字母开头,`json.marshal` 将无法正确导出这些字段的值,导致输出的 json 对象为空。本文将深入探讨 go 语言的字段可见性规则,并通过示例代码演示如何通过将结构体字段的首字母大写来使其可被导出,从而确保 `json.marshal` 能够成功序列化结构体的完整内容。

Go 语言中 json.Marshal 的常见陷阱

Go 语言的 encoding/json 包提供了一套强大且灵活的机制,用于在 Go 值和 JSON 格式之间进行转换。然而,初学者在使用 json.Marshal 将自定义结构体序列化为 JSON 字符串时,常常会遇到一个令人困惑的问题:尽管结构体实例中明明包含了数据,但序列化后的 JSON 字符串却显示为空对象(例如 {"key":{}})。这通常不是因为 json.Marshal 本身存在错误,而是源于 Go 语言中一个核心概念——字段的可见性(或称导出性)。

问题现象分析

考虑以下 Go 代码示例,其中定义了一个名为 node 的结构体,其所有字段都以小写字母开头:

package main 

import (
    "encoding/json"
    "fmt"
)

type node struct {
    value   string
    expiry  float64
    settime float64
}

func main() {
    var x = make(map[string]node)

    x["hello"] = node{value: "world", expiry: 1, settime: 2}
    x["foo"] = node{value: "bar", expiry: 1, settime: 2}

    a, err := json.Marshal(x)
    if err != nil {
        fmt.Println("Error marshalling:", err)
        return
    }
    fmt.Println(string(a))
}

运行上述代码,预期会得到包含 value、expiry 和 settime 字段的 JSON 输出。然而,实际输出却是:

{"foo":{},"hello":{}}

可以看到,"foo" 和 "hello" 键对应的值都是空的 JSON 对象 {},结构体 node 内部的字段值并未被序列化。

根本原因:Go 语言的导出规则

在 Go 语言中,标识符(包括结构体字段、函数、方法、变量等)的可见性是由其首字母的大小写决定的:

  • 大写字母开头的标识符:表示该标识符是导出(exported)的,可以在包外部访问。
  • 小写字母开头的标识符:表示该标识符是非导出(unexported)的,只能在其所在的包内部访问。

encoding/json 包在执行 json.Marshal 操作时,只会序列化结构体中导出的字段。对于非导出字段,它会将其忽略。在上述示例中,node 结构体中的 value、expiry 和 settime 字段都以小写字母开头,因此它们是非导出的,json.Marshal 无法访问并序列化它们。

TapNow
TapNow

新一代AI视觉创作引擎

下载

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

解决这个问题的关键在于遵循 Go 语言的导出规则,将需要序列化到 JSON 的结构体字段的首字母改为大写。

修改结构体定义

将 node 结构体修改为 Node(通常结构体名称也建议首字母大写以表示其可导出),并将其字段的首字母也改为大写:

package main 

import (
    "encoding/json"
    "fmt"
)

// Node 结构体,字段首字母大写,表示可导出
type Node struct {
    Value   string
    Expiry  float64
    Settime float64
}

func main() {
    var x = make(map[string]Node) // 注意这里也需要使用 Node 类型

    x["hello"] = Node{Value: "world", Expiry: 1, Settime: 2}
    x["foo"] = Node{Value: "bar", Expiry: 1, Settime: 2}

    a, err := json.Marshal(x)
    if err != nil {
        fmt.Println("Error marshalling:", err)
        return
    }
    fmt.Println(string(a))
}

预期输出

现在,当运行修改后的代码时,json.Marshal 将能够正确识别并序列化 Node 结构体的所有导出字段,输出结果将是:

{"foo":{"Value":"bar","Expiry":1,"Settime":2},"hello":{"Value":"world","Expiry":1,"Settime":2}}

这正是我们期望的完整 JSON 序列化结果。

总结与注意事项

  1. 导出规则是核心:在 Go 语言中,任何需要被 encoding/json 包序列化的结构体字段都必须是导出的(即首字母大写)。
  2. 不仅仅是 json.Marshal:Go 语言的导出规则适用于整个 Go 生态系统,不仅限于 encoding/json。它影响着包之间的可见性、接口实现等多个方面。
  3. json 标签的用途:如果希望 JSON 字段名与 Go 结构体字段名不同,或者需要对字段进行其他控制(例如忽略空值),可以使用结构体字段标签(json:"field_name,omitempty")。但即使使用了标签,字段本身也必须是导出的。例如:
    type User struct {
        Name string `json:"user_name"` // 导出字段,JSON名为 user_name
        Age  int    `json:"age,omitempty"` // 导出字段,如果为零值则在JSON中忽略
    }
  4. json.Unmarshal 同样适用:在将 JSON 数据反序列化到 Go 结构体时(使用 json.Unmarshal),目标结构体的字段也必须是导出的,才能接收对应的 JSON 值。
  5. 错误处理:在实际开发中,务必对 json.Marshal 和 json.Unmarshal 的返回错误进行妥善处理,以便及时发现并解决序列化/反序列化过程中可能出现的问题。

理解并正确应用 Go 语言的导出规则是有效使用 encoding/json 包的关键。通过确保结构体字段的首字母大写,可以避免常见的序列化问题,并确保 Go 应用程序与 JSON 数据之间的数据交换顺畅无误。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

411

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

533

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

309

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

74

2025.09.10

mysql标识符无效错误怎么解决
mysql标识符无效错误怎么解决

mysql标识符无效错误的解决办法:1、检查标识符是否被其他表或数据库使用;2、检查标识符是否包含特殊字符;3、使用引号包裹标识符;4、使用反引号包裹标识符;5、检查MySQL的配置文件等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

180

2023.12.04

Python标识符有哪些
Python标识符有哪些

Python标识符有变量标识符、函数标识符、类标识符、模块标识符、下划线开头的标识符、双下划线开头、双下划线结尾的标识符、整型标识符、浮点型标识符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

278

2024.02.23

java标识符合集
java标识符合集

本专题整合了java标识符相关内容,想了解更多详细内容,请阅读下面的文章。

253

2025.06.11

c++标识符介绍
c++标识符介绍

本专题整合了c++标识符相关内容,阅读专题下面的文章了解更多详细内容。

121

2025.08.07

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

9

2026.01.16

热门下载

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

精品课程

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

共101课时 | 8.3万人学习

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号