Go指针在结构体中应该如何使用_Go Pointer结构体绑定说明

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 值也能调用(只要方法内不解引用该指针),这可以用来表示“空状态”,但需主动防御。

基本上就这些。用不用指针,核心看两点:要不要改原值?要不要避免拷贝?想清楚这两点,结构体里的指针就用得自然又安全。