

新闻资讯
技术学院必须用指针的场景有五种:一是函数内修改结构体字段并需影响原值;二是结构体较大时避免拷贝;三是方法接收者为* T 时调用对象须可寻址;四是初始化需获取指针;五是反射操作前需 Elem() 获取可设置值。
当你需要在函数内修改结构体字段,并让调用方看到变更,就必须传入结构体指针。传值会复制整个结构体,函数内修改只作用于副本。func updateName(p *Person) 能改原值;func updateName(p Person) 不能。
*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
if p == nil,而不是 if *p == (Person{})
反射操作结构体字段时,若传入的是指针,需先用 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 或需共享/延迟初始化的字段,用指针更灵活;对小而确定存在的字段,用值类型更安全简洁。
type Address struct {
City string
Zip string
}
type Profile struct {
Name string // 值类型:轻量、必有
Avatar *string // 指针:可能未上传,nil 表示“无头像”
Location *Address //
指针:避免复制,且允许为 nil 表示“地址未填写”
}
*string 字段为 null 会得到 nil;string 字段为 null 会得到空字符串,语义不同Profile{Location: &Address{}} 是合法的,但 Profile{Location: &Address} 语法错误——&Address 不是类型,应为 *Address