必须用指针的场景有五种:一是函数内修改结构体字段并需影响原值;二是结构体较大时避免拷贝;三是方法接收者为* T 时调用对象须可寻址;四是初始化需获取指针;五是反射操作前需 Elem() 获取可设置值。

结构体字段赋值时,什么时候必须用指针
当你需要在函数内修改结构体字段,并让调用方看到变更,就必须传入结构体指针。传值会复制整个结构体,函数内修改只作用于副本。func updateName(p *Person) 能改原值;func updateName(p Person) 不能。
- 结构体较大(如含切片、map、大量字段)时,传指针避免不必要的内存拷贝
- 方法接收者声明为
*T时,调用该方法的对象必须可寻址(即不能是字面量或临时值),例如(&Person{}).Method()合法,Person{}.Method()编译报错:cannot call pointer method on Person literal - 初始化结构体时,
&Person{Name: "Alice"}得到指针;Person{Name: "Alice"}得到值
访问结构体指针字段的两种写法等价吗
等价。Go 允许对结构体指针直接使用点号访问字段,无需显式解引用。p.Name 和 (*p).Name 效果完全相同,编译器自动处理。
type Person struct {
Name string
Age int
}
p := &Person{Name: "Bob", Age: 30}
p.Name = "Tom" // ✅ 合法且推荐
(*p).Age = 31 // ✅ 合法但冗余
- 字段赋值、方法调用、嵌套结构体访问都支持这种隐式解引用
- 若指针为
nil,访问字段会 panic:panic: runtime error: invalid memory address or nil pointer dereference - 判断是否为 nil 应写成
if p == nil,而不是if *p == (Person{})
结构体字段标签(tag)和反射配合指针使用的注意事项
反射操作结构体字段时,若传入的是指针,需先用 reflect.Value.Elem() 获取其指向的值,否则无法读写导出字段。
type User struct {
ID int `json:"id"`
Name string `json:"name"`
}
u := &User{ID: 123, Name: "Lee"}
v := reflect.ValueOf(u)
if v.Kind() == reflect.Ptr {
v = v.Elem() // 必须这一步,否则 v.Field(0) 会 panic
}
v.Field(0).SetInt(456) // 修改 ID
- 反射对象必须可设置(
v.CanSet() == true),而传值时reflect.ValueOf(User{})的字段不可设置,只有指针才可能可设置 - 字段名必须首字母大写(导出),否则反射无法访问
- 常见错误:忘记
.Elem(),导致Field(i)报错panic: reflect: Field of non-struct type
嵌套结构体中混合使用值类型和指针字段的典型场景
嵌套结构体里,对可能为 nil 或需共享/延迟初始化的字段,用指针更灵活;对小而确定存在的字段,用值类型更安全简洁。
立即学习“go语言免费学习笔记(深入)”;
type Address struct {
City string
Zip string
}
type Profile struct {
Name string // 值类型:轻量、必有
Avatar *string // 指针:可能未上传,nil 表示“无头像”
Location *Address // 指针:避免复制,且允许为 nil 表示“地址未填写”
}
- JSON 反序列化时,
*string字段为 null 会得到 nil;string字段为 null 会得到空字符串,语义不同 - 数据库 ORM(如 GORM)常将外键关联字段设为指针,以区分“未设置”和“设为空字符串/零值”
- 注意:
Profile{Location: &Address{}}是合法的,但Profile{Location: &Address}语法错误——&Address不是类型,应为*Address










