贝利信息

Golang函数参数是值传递还是引用传递_参数传递机制详解

日期:2026-01-13 00:00 / 作者:P粉602998670
Go函数参数均为值传递,slice/map/chan因底层含指针字段,修改其内容可影响原变量;仅当需修改变量本身或规避大对象拷贝时才用指针传参。

Go 语言中所有函数参数都是值传递——包括 slicemapchanfunc*T 等看似“引用类型”的参数,传的仍是该值的副本。但因为这些类型的底层结构包含指针字段,所以修改其内部数据可能影响原变量,容易误以为是引用传递。

为什么 slice 传参后能修改底层数组?

slice 是一个三字段的结构体:struct{ ptr *T; len, cap int }。值传递时,这三个字段被完整复制,其中 ptr 字段是地址,指向同一块底层数组。因此:

func modify(s []int) {
    s[0] = 999        // ✅ 影响原 slice
    s = append(s, 1)  // ❌ 不影响调用方的 s(除非扩容未发生且你观察的是同一底层数组)
}
func main() {
    a := []int{1, 2, 3}
    modify(a)
    fmt.Println(a[0]) // 输出 999
}

map 和 chan 为什么也“像引用传递”?

mapchan 类型的底层是运行时分配的指针(如 *hmap*hchan),它们的变量本身存储的就是这个指针的值。值传递时,复制的是该指针值,所以多个变量指向同一运行时结构:

注意:nil mapnil chan 传参后仍为 nil,不能直接写入,否则 panic。

什么时候必须用指针传参?

只有两类情况真正需要 *T 参数:

常见误用:对小结构体(如 type Point struct{ X, Y int })盲目加 *,反而增加解引用开销;对只读 stringint[32]byte 等传指针纯属多余。

func updateName(p *Person) {
    p.Name = "Alice" // ✅ 修改调用方的 Person 实例
}
func copyName(p Person) {
    p.Name = "Bob" // ❌ 只改副本,原变量不变
}

容易忽略的关键细节

最常被忽视的不是“怎么传”,而是“谁 owns 底层资源”。例如:

判断是否要传指针,别看类型名,要看你是否需要改变调用方变量所指向的内存位置,或者是否在规避可观的拷贝成本。