Go中JSON序列化时值类型与指针类型行为一致,因json.Marshal会自动解引用指针;但nil指针序列化为null,而零值字段使用默认值,如空字符串或0;反序列化时指针可区分字段是否提供,配合omitempty能判断字段是否存在,嵌套指针字段可自动分配内存;因此对需区分“未设置”与“零值”的场景,使用指针更灵活。

在Go语言中,JSON序列化和反序列化由
encoding/json包处理。值类型和指针类型在这过程中表现不同,但大多数情况下行为一致,关键在于字段是否存在以及字段的可导出性(首字母大写),而不是是否为指针。
值类型与指针类型的序列化(Marshal)
序列化时,
json.Marshal会自动解引用指针字段,因此值类型和其对应指针类型生成的JSON内容通常相同。 示例:
定义结构体:
type Person struct {
Name string `json:"name"`
Age int `json:"age"`
}
分别使用值和指针:
立即学习“go语言免费学习笔记(深入)”;
personVal := Person{Name: "Alice", Age: 30}
personPtr := &personVal
jsonVal, _ := json.Marshal(personVal)
jsonPtr, _ := json.Marshal(personPtr)
// 输出都是:{"name":"Alice","age":30}
可见,指针在序列化时会被自动解引用,结果一致。
零值与nil指针的差异
当指针为
nil时,序列化结果会变成
null,而值类型总是使用其零值。
例如:
type Profile struct {
Nickname *string `json:"nickname"`
}
var nickname *string = nil
profile := Profile{Nickname: nickname}
json.Marshal(profile) // 输出:{"nickname":null}
如果字段是值类型
Nickname string,则输出为
""(空字符串)。
这种差异在API设计中很重要,
null可表示“未设置”,而空字符串表示“显式为空”。
反序列化(Unmarshal)时的指针优势
反序列化时,使用指针类型可以判断字段是否在JSON中出现,尤其是配合
omitempty时。
例如:
type User struct {
Name string `json:"name"`
Age *int `json:"age,omitempty"`
}
如果JSON中没有
age字段,
age字段将为
nil;如果有,则指向具体值。
这允许你区分:
age
字段未提供(nil
)age
字段为0(new(int)
,值为0)
而如果
Age int是值类型,无法判断是默认零值还是JSON中提供了0。
嵌套结构体与指针字段
嵌套字段为指针时,反序列化能正确分配内存并填充数据。
例如:
type Address struct {
City string `json:"city"`
}
type Person struct {
Name string `json:"name"`
Address *Address `json:"address,omitempty"`
}
即使
Address字段是
*Address,只要JSON中有
Address对象,
json.Unmarshal会自动创建
Address实例并赋值给指针。
若JSON中无该字段,指针保持
nil,可用于判断是否存在地址信息。
基本上就这些。Go的
json包对指针处理很友好,关键在于理解
nil和零值的语义差异,以及如何利用指针实现“可选字段”的精确表达。实际开发中,对可能缺失或需要区分“未设置”和“零值”的字段,使用指针类型更灵活。










