Go中struct是唯一基础单元,字段首字母大小写决定导出性,推荐字段名赋值初始化,匿名字段实现组合式字段提升,方法接收者选值或指针取决于是否修改原值,私有字段不参与JSON序列化和ORM映射。
Go 语言中没有“类”,struct 就是你组织数据的唯一基础单元——它不是语法糖,而是内存布局、序列化、方法绑定的共同载体。定义和使用看似简单,但字段可见性、初始化顺序、指针语义这些细节一旦出错,轻则字段访问失败,重则序列化丢数据、方法不生效。
结构体字段是否导出(即能否被其他包访问),只看首字母大小写,和是否加 public 或 private 关键字完全无关:
Name string → 导出字段,外部包可读写age int → 未导出字段,仅当前包内可用(哪怕你加了注释或 tag,外部也无法访问)Name: string(那是 map 初始化语法)type User struct{ ID int } 和 type Order struct{ ID int } 完全合法)type User struct {
ID int `json:"id"`
Name string `json:"name"`
email string // 小写,外部包无法访问
}
初始化不是“能跑就行”,它直接影响可读性、字段遗漏风险和后续扩展成本:
u := User{ID: 123, Name: "Alice"},未填字段自动为零值(email 为空字符串)u := User{123, "Alice", "a@example.com"};一旦结构体新增字段或调整顺序,所有这类初始化都会静默错位u := &User{ID: 123} 更直观;u := new(User) 返回 *User,但所有字段仍为零值,需手动赋值,无明显优势Go 不支持继承,但通过匿名字段实现“字段提升”(field promotion),这是组合式设计的关键:
Addr Address → 访问需写 u.Addr.City
Address(无字段名)→ u.City 和 u.Address.City 都合法;若冲突(比如 User 和 Address 都有 City),优先使用直接字段(u.City)Person),不能是基础类型(int、string)或未命名结构体type Address struct { City, Zip string }
type User struct {
Name string
Address // 匿名字段
}
u := User{Name: "Bob", Address: Address{City: "Shanghai"}}
fmt.Println(u.City) // 输出 "Shanghai",字段被提升
接收者类型不是性能优化题,而是语义题:你写的这个方法,是否需要修改调用者的原始数据?
func (u User) SetName(n string) → 方法内对 u.Name 赋值,不影响原始变量(传的是副本)func (u *User) SetEmail(e string) → 可修改原始 User 实例的字段,且调用时 u.SetEmail("x@y.z") 和 (&u).SetEmail("x@y.z"
) 都合法(Go 自动取址)最容易被忽略的是:结构体字段的可见性在反射、JSON 序列化、数据库 ORM 中全部生效——私有字段(小写)不会被 json.Marshal 输出,也不会被 gorm 映射到数据库列。别等上线后发现 API 返回空字段才回头翻文档。