Go中指针是控制数据所有权和共享行为的基本工具:值传递复制数据,指针传递共享内存;结构体传参、方法接收者、接口实现及nil判空均依赖此本质区别。
Go 中指针不是“高级技巧”,而是控制数据所有权和共享行为的基本工具;普通变量存值,指针存地址——这个区别直接决定函数能否修改原变量、结构体传参是否卡顿、方法调用是否 panic。
modify(x int) 改不了原值,但 modify(&x) 可以Go 所有参数传递都是值传递,关键在于“传的是什么的值”:
modify(x int):传的是 x 的副本,函数内改 x = 改一张复印件,原变量毫发无损modify(p *int):传的是地址的副本(比如 0xc000012340),虽是地址的拷贝,但两个地址指向同一块内存,*p = 100 就是往那块内存写入常见错误现象:func setName(p Person) { p.Name = "Alice" } 调用后 Person 原值没变——因为传的是整个 struct 的拷贝,不是地址。
假设有个 type BigData struct { Items [1e6]int }(约 8MB):
process(b BigData) → 每次调用都复制 8MB 内存,GC 压力陡增,延迟明显process(b *BigData) → 只传一个 8 字节地址,开销恒定实操建议:只要 struct 字段超过 4–5 个基础类型,或含 slice/map/channel,优先用 *T 传参;标准库中 http.Request、sql.Rows 全部是指针类型,不是巧合。
func (t T) M() 还是 func (t *T) M()?这不是风格问题,而是接口实现和可变性的硬约束:
t.Counter++),必须用指针接收者 *T,否则改的是副本*T 类型能赋值给该接口,T 类型会编译报错:T does not implement X (M method has pointer receiver)
*T 接收者——避免后续加修改逻辑时被迫重构所有调用点典型坑:var u User; fmt.Printf("%v", u) 没问题,但 var i fmt.Stringer = u 编译失败,只因 User.String() 是指针接收者。
nil,值类型的零值是具体值——这影响初始化和判空逻辑这是最易被忽略的语义差异:
var s string → s == "",安全使用var p *string → p == nil,解引用前必须检查:if p != nil { use(*p) },否则 panic: invalid memory address or nil pointer dereference
type Config struct { DBURL *string }),JSON 反序列化可能留 nil,直接 *c.DBURL 会 crash真正复杂的地方不在语法,而在于:你得时刻判断——这个变量,我到底想让它「独立」还是「共享」?想隔离就用值,想联动就用指针;想省事就用值,想高效就用指针;但一旦混用,bug 往往出现在边界处,比如 nil 指针解引用、接口赋值失败、或以为改了实则没改。