Go中嵌套结构体字段可链式访问:顶层指针自动解引用,值类型字段直接访问,指针字段自动解一次;修改需确保可寻址性,JSON解析后指针字段为nil需判空,反射访问要防nil panic。
Go 中只要顶层变量是指针,后续所有嵌套字段(无论是否为指针)都能用 . 直接访问,编译器自动解引用。不需要手动写 *p.Field.SubField 这种冗余形式。
常见错误是误以为“指针的指针”才需要多次解引用——其实只要字段本身不是指针类型,p.Field.SubField 就合法且高效。
Person 是结构体,Address 是其嵌套结构体字段 → personPtr.Address.City 可直接读写Address 是 *Address 指针字段 → 仍可写 personPtr.Address.City,Go 自动解一次 *
City 是 *string,才需显式解引用:*personPtr.Address.City
type Address struct {
City string
}
type Person struct {
Name string
Addr Address // 值类型字段
AddrP *Address // 指针类型字段
}
func main() {
p := &Person{Name: "Alice", Addr: Address{City: "Beijing"}}
fmt.Println(p.Addr.City) // ✅ 合法:自动解 p 的指针,再取 Addr.City
p.AddrP = &Address{City: "Shanghai"}
fmt.Println(p.AddrP.City) // ✅ 合法:自动解 p,再自动解 AddrP,再取 City
}
如果函数想修改嵌套结构体中的某个字段,必须确保该字段所在的结构体实例是可寻址的(即传入的是指针),否则会编译失败。
典型报错:cannot assign to struct field xxx.yyy in map or slice 或 cannot assign to s.Field,往往是因为你从 map/slice 中取出来的值是副本,不可寻址。
map[string]Person 取出的 Person 是副本 → m["a"].Name = "Bob" 报错personPtr := &m["a"] → 然后 personPtr.Addr.City = "Guangzhou"

s[0].Field = x,应改用 &s[0] 获取地址再操作用 reflect 动态访问嵌套字段时,如果中间某层是 nil 指针(如 Person.AddrP 为 nil),调用 FieldByName("City") 会 panic。
必须逐层检查是否为 nil,或用 reflect.Value.Elem() 前确认 CanInterface() 和非空。
reflect.Indirect() 处理可能为 nil 的指针if v.Kind() == reflect.Ptr && v.IsNil() { ... }
用 json.Unmarshal 解析时,如果 JSON 中缺失某个嵌套字段,对应 Go 字段行为取决于它是值类型还是指针类型:
Address)会被初始化为零值(Address{})*Address)保持 nil,不会自动分配内存p.AddrP.City 前必须判空,否则 panicvar p Person
json.Unmarshal([]byte(`{"Name":"Tom"}`), &p)
// p.Addr 是 {}(零值),p.AddrP 是 nil
fmt.Println(p.Addr.City) // ✅ 输出空字符串
fmt.Println(p.AddrP.City) // ❌ panic: invalid memory address or nil pointer dereference
嵌套字段操作最常被忽略的点是:**字段是否可寻址**——它不只影响赋值,还决定能否取地址、能否用反射修改、能否作为方法接收者调用。很多 runtime panic 都源于假设了一个不可寻址的临时值能被修改。