Go函数参数均为值传递,传值类型改副本不影响原变量,传指针类型可通过解引用修改原内存;结构体依大小和修改需求选传值或指针;切片/map/channel属“假引用”,可改元素但不可扩容;方法接收者依是否修改状态选择值或指针。

函数里改不了变量?先看传的是值还是地址
Go 里所有函数参数都是值传递,但“值”本身可能是整数,也可能是内存地址——关键就在这里。如果你发现 modifyName(p Person) 没改成功原始变量,大概率是因为你传了值;而换成 modifyName(p *Person) 并传入 &person,就能生效。
- 值类型(
int、string、小struct)传参:函数拿到的是副本,改了不影响原变量 - 指针类型(
*int、*Person)传参:函数拿到的是地址副本,通过*p解引用就能写原内存 - 常见错觉:“Go 支持引用传递”——其实没有,只是你传了个地址值而已
结构体该传值还是传指针?看大小和是否要改
一个 type User struct { Name string; Email string } 只有两个字段,传值没问题;但要是加了 AvatarData [1024 * 1024]byte,每次调用都复制 1MB,性能立刻掉下来。
- 小结构体(≤ 3 字段,无大数组/切片):传值更安全,语义清晰,避免意外修改
- 大结构体或含大字段(如图片 buffer、配置 map):必须传指针,否则拷贝开销不可接受
- 只要有一个方法用了指针接收者(比如
func (u *User) Save()),整个类型建议统一用指针接收者,否则接口赋值会失败
切片、map、channel 是“假引用”,别被名字骗了
它们底层确实含指针(比如 slice 有 data *byte),所以你在函数里改 s[0] = 10 能看到效果;但想扩容或换底层数组?不行——因为传参仍是值传递,只复制了 header 结构。
- 能改元素内容:✅
s[0] = 10、m["k"] = v都生效 - 不能改长度/容量/底层数组:❌
s = append(s, x)不影响原切片;必须用func extend(s *[]int, x int)+*s = append(*s, x) - nil map 或 nil slice 解引用会 panic,但 nil 指针解引用也会 panic——这点一致,别以为“引用类型就更安全”
方法接收者选值还是指针?看它要不要改状态
Go 的方法集规则很严格:type T 的方法集只包含定义在 T 上的方法;*T 的方法集包含 T 和 *T 上的所有方法。这意味着:如果你写了 func (u *User) SetEmail(e string),那只有 *User 能调用它,User 值类型直接不满足接口。
立即学习“go语言免费学习笔记(深入)”;
- 需要修改字段(
Inc()、Reset()、SetXXX())→ 必须用指针接收者 - 纯读取(
FullName()、IsValid())→ 值接收者更轻量、更安全 - 实际项目中,只要结构体稍大或未来可能加修改方法,就直接上
*T,省得后期重构接口和调用点
最容易被忽略的一点:不是“指针更高级”,而是“你传的值是否指向原始数据”。 写 fmt.Println(&x) 看地址,写 fmt.Println(*p) 看内容,两步拆开理解,比背概念管用得多。









