Go函数参数均为值传递,无法直接修改原始变量;需修改时必须传指针,且调用时显式取地址并判空;切片、map等虽可修改底层数据,但仍是值类型,替换整体需指针或返回新值。

Go 函数参数是值拷贝,无法直接修改原始变量
Go 中所有函数参数都是按值传递(pass by value),int、string、struct 等值类型传入函数时,实际上传的是副本。函数内对形参的任何赋值操作,都不会影响调用方的原始变量。
想修改原始值,必须传指针
要让函数能修改调用方的值类型变量,唯一可靠方式是传入该类型的指针。函数通过解引用(*p)写入新值,才能反映到原变量上。
-
func increment(x *int) { *x = *x + 1 }—— 正确:通过指针修改原始int -
func increment(x int) { x = x + 1 }—— 错误:只改了副本,调用方无感知 - 调用时需显式取地址:
increment(&a),不能传increment(a) - 若传
nil指针,解引用会 panic,务必先判空(尤其在可选参数或配置场景中)
结构体传参:值拷贝 vs 指针性能与语义差异
小结构体(如 type Point struct{ X, Y int })按值传参开销小且安全;大结构体(含切片、map 或大量字段)按值传会复制整块内存,既慢又浪费。更重要的是——语义上是否需要「修改原始实例」。
- 只读访问:传值更清晰、无副作用,编译器也更容易优化
- 需修改字段:必须传
*MyStruct,否则字段赋值无效 - 方法接收者用指针(
func (s *MyStruct) SetX(x int))才可能修改原结构体;值接收者(func (s MyStruct) SetX(x int))同理只改副本
常见误判场景:切片、map、channel 是引用类型但不是“引用传递”
虽然 []int、map[string]int、chan int 在函数内能修改其底层数据(如追加元素、增删键),但这不等于它们是“引用传递”。它们本身仍是值类型——只是其底层结构包含指针字段。所以:
立即学习“go语言免费学习笔记(深入)”;
- 可以修改底层数组/哈希表内容,但无法让调用方变量指向新底层数组(例如
s = append(s, 1)不会影响原切片头) - 若需替换整个切片头(如扩容后地址变更),仍需传
*[]int - 错误示例:
func badAppend(s []int) { s = append(s, 99) } // 调用后原切片长度不变,append 的新底层数组未返回 - 正确做法:
func goodAppend(s []int) []int { return append(s, 99) } // 或 func goodAppendInPlace(s *[]int) { *s = append(*s, 99) }
time.Time 这样的“看起来像引用”的类型,它也是纯值类型,修改必须靠返回新值或指针。










