贝利信息

Golang结构体字段使用指针的设计考量

日期:2026-01-09 00:00 / 作者:P粉602998670
结构体字段应声明为指针当且仅当需表达可空性、延迟初始化、共享修改或避免大对象拷贝;小值类型若确定非空且无需区分零值与未设置,则优先用值类型。

什么时候该把结构体字段声明为指针

字段用指针不是为了“看起来高级”,而是明确表达「该字段可为空、可延迟初始化、或需共享修改」。如果字段类型是 stringintbool 这类小值类型,又确定非空且不需区分零值与未设置,用值类型更安全、更直观。

JSON 反序列化时 *string 字段为何常为空

Go 的 encoding/json 在遇到 JSON 中缺失字段或 null 值时,会把 *string 字段设为 nil;但如果字段是 string,则设为 ""。这容易让人误以为“没反序列化成功”,其实只是符合预期行为。

方法接收者用指针时,字段指针是否必须同步调整

无关。接收者是否用指针(func (s *MyStruct) Do())只影响结构体本身能否被修改,和其内部字段是否为指针完全解耦。字段指针的设计决策应独立于接收者类型。

type Config struct {
    TimeoutSec *int    `json:"timeout_sec,omitempty"`
    LogPath    *string `json:"log_path,omitempty"`
}

// 反序列化时:
// { "timeout_sec": 30 } → TimeoutSec 指向 int(30)
// { "log_path": null }  → LogPath == nil
// { }                   → TimeoutSec == nil, LogPath == nil
字段指针最易被忽略的点不在语法,而在所有权和生命周期:谁负责分配?谁负责释放?是否可能悬空?这些问题不厘清,*T 就只是把 bug 从编译期推迟到运行时。