Go中解引用是用*符号通过指针地址访问或修改所指向的值,仅适用于指针类型,结构体指针字段访问自动解引用,但需检查非nil以避免panic。
Go 中的解引用操作,就是用 * 符号“打开”指针,拿到它背后指向的那个值。它不是魔法,而是一次明确的内存访问:指针里存的是地址,* 就是按这个地址去把数据取出来。
解引用的基本写法
声明一个变量,再用 & 取它的地址得到指针,最后用 * 解引用:
- var num int = 42 —— 普通整数
- ptr := &num —— ptr 是 *int 类型,值是 num 的内存地址
- value := *ptr —— 解引用,value 等于 42,类型是 int
注意:* 只能作用于指针类型。对非指针变量(比如直接写 *num)会编译报错。
解引用不只是读,还能写
解引用后得到的是原
变量的“替身”,可以直接赋值,从而修改原始值:
- *ptr = 100 —— 这行代码等价于 num = 100
- 后续再读 num 或 *ptr,都得到 100
这是指针最实用的地方:让函数能真正修改外部变量,而不是只改副本。
结构体指针不用显式解引用
对结构体指针,Go 提供了语法糖:访问字段或调用方法时,自动帮你解引用,不需要写 (*ptr).Field:
- type User struct { Name string }
- u := &User{Name: "Tom"}
- fmt.Println(u.Name) ✅ 正确,Go 自动转为 (*u).Name
- fmt.Println(*u.Name) ❌ 错误,u.Name 已经是 string 类型,不能再加 *
安全解引用必须先判空
nil 指针解引用会 panic,运行时报错 “invalid memory address or nil pointer dereference”。所以实际开发中要养成习惯:
- 用前检查:if ptr != nil { value := *ptr }
- 或者封装成安全函数,比如泛型版:func Deref[T any](p *T, def T) T { if p == nil { return def }; return *p }
基本规则就一条:准备写 * 的那一刻,心里默念——这个指针,它真的指向有效内存吗?
基本上就这些。








