Go中结构体指针主要用于避免大对象复制、实现可变性及支持方法接收者绑定;字段用指针适用于共享、延迟初始化或可选场景,方法接收者用指针才能修改原结构体。

Go 中结构体里的指针主要用于避免复制大对象、实现可变性、以及支持方法接收者绑定。关键不是“能不能用”,而是“什么时候该用”。
结构体字段中存指针:为共享或延迟初始化
当某个字段值较大(如大 slice、map、嵌套结构体),或需要被多个结构体实例共享,或允许为 nil 表示“未设置/可选”,就适合用指针类型字段。
例如:
type User struct {
Name string
Avatar *Image // 可能很大,且不总是存在
Profile *UserProfile // 可选扩展信息
}
- 赋值时不会复制 *Image 指向的数据,只复制指针本身(8 字节)
- 多个 User 可以共用同一张头像,修改一处,所有引用可见
- 初始化时可设为 nil,后续按需 new 或赋值
方法接收者用指针:要修改结构体自身
如果方法内部要修改结构体的字段,接收者必须是指针类型。值接收者操作的是副本,改了也白改。
func (u *User) SetName(name string) {
u.Name = name // ✅ 正确:修改原结构体
}
func (u User) SetNameBad(name string) {
u.Name = name // ❌ 无效:只改了副本
}
即使没显式修改字段,只要方法签名用了指针接收者,调用时 Go 会自动取地址——所以 user.SetName("Alice") 在 user 是变量时也能工作。
创建结构体时如何传指针
常见方式有三种,语义略有不同:
- &User{Name: "Tom"} —— 创建匿名结构体再取地址,最常用
- new(User) —— 返回指向零值结构体的指针,字段全为零(Name="")
- 先声明变量再取址:u := User{Name: "Tom"}; ptr := &u —— 适合需要复用变量名或后续多次赋值的场景
注意 nil 指针解引用 panic
结构体字段是指针时,访问前最好判空,尤其来自 JSON 解析、数据库查询等外部输入:
if u.Avatar != nil {
fmt.Println(u.Avatar.URL)
} else {
fmt.Println("no avatar")
}
方法接收者为指针时,nil 值也能调用(只要方法内不解引用该指针),这可以用来表示“空状态”,但需主动防御。
基本上就这些。用不用指针,核心看两点:要不要改原值?要不要避免拷贝?想清楚这两点,结构体里的指针就用得自然又安全。










