Go函数参数永远是值传递:值类型传副本,指针传地址值;切片/map/channel是含指针的结构体,表现类似引用;大对象或需修改原变量时用指针。

Go 语言中,函数参数传递永远是值传递,但“值”的内容取决于你传的是变量本身(值类型)还是它的地址(指针)。理解这一点,就抓住了 Go 指针与值关系的核心。
值类型传参:复制一份数据
int、float64、string、struct、array 等属于值类型。传入函数时,Go 复制整个值的副本,函数内修改不影响原变量。
例如:
func modify(x int) {
x = 100 // 只改了副本
}
n := 42
modify(n)
fmt.Println(n) // 还是 42这种机制安全、直观,但对大 struct 或数组来说,复制开销明显。
立即学习“go语言免费学习笔记(深入)”;
指针传参:传递地址,可间接修改原值
用 * 声明指针变量,用 & 取地址。传指针,等于把变量的内存地址“值”传进去——函数拿到地址后,就能通过 * 解引用去读写原位置的数据。
例如:
func modifyPtr(x *int) {
*x = 100 // 修改地址指向的值
}
n := 42
modifyPtr(&n)
fmt.Println(n) // 输出 100注意:&n 是一个 *int 类型的值(即地址),它本身仍是被复制传递的——只是这个“值”恰好能定位到原变量。
切片、map、channel 是引用类型?其实是“带指针的结构体”
它们不是真正的引用类型(Go 没有引用类型),而是由多个字段组成的结构体,其中包含指向底层数据的指针。所以传参时虽是值传递,但结构体里的指针字段被复制了,新旧变量仍指向同一块底层数组或哈希表。
- 切片:含 ptr(指向底层数组)、len、cap —— 修改元素会反映到原切片;但 append 若扩容,可能生成新底层数组,原切片不受影响
- map 和 channel:内部含指针,所以函数内增删元素会影响原 map/channel
因此它们“表现得像引用”,但本质仍是值传递 + 内部指针共享。
什么时候该用指针?记住两个关键场景
- 需要函数修改调用方的变量值(如初始化结构体字段、解析结果写回)
- 避免大对象(比如 > 128 字节的 struct)频繁复制,提升性能和内存效率
- 实现接口时,方法集与接收者类型有关:只有 *T 方法集能同时被 T 和 *T 调用;而 T 方法集只能被 T 调用(除非 T 是可寻址的)
小结构体(如 type Point struct{ X, Y int })通常直接传值更清晰;含大量字段或需统一修改语义时,优先用指针接收者。
基本上就这些。Go 不搞隐式引用,所有指针行为都显式可见——& 取地址,* 解引用,传什么、改什么,一目了然。










