登录
首页 >  Golang >  Go教程

Golang值传递原理与使用技巧

时间:2026-01-16 18:39:39 491浏览 收藏

小伙伴们对Golang编程感兴趣吗?是否正在学习相关知识点?如果是,那么本文《Golang值传递机制详解》,就很适合你,本篇文章讲解的知识点主要包括。在之后的文章中也会多多分享相关知识点,希望对大家的知识积累有所帮助!

Go函数参数均为值传递,slice/map/channel/interface传的是含指针的结构体副本,故修改元素或写入键值会影响原变量,但重赋值不会;需替换整个结构时须显式传指针。

如何在Golang中理解值类型传递_Golang函数参数传值机制

Go 函数参数永远是值传递,包括 slice、map、channel、interface

Go 语言没有“引用传递”,所有函数调用都是值传递。但关键在于:传的是「什么的值」。比如 slice 类型本身是一个结构体(含指针、len、cap),传参时复制的是这个结构体,不是底层数组;mapchannel 同理,底层是运行时管理的指针封装。所以修改 slice 元素或向 map 写入键值,会影响原变量;但对形参重新赋值(如 s = append(s, x)m = make(map[int]int))不会影响实参。

什么时候修改形参会反映到实参,什么时候不会

判断依据是:操作是否通过形参持有的指针间接修改了共享的底层数据。

  • ✅ 会反映:修改 slice[i]、调用 map[k] = v、向 channel 发送/接收、调用 interface{} 方法(若方法集含指针接收者且原值可寻址)
  • ❌ 不会反映:s = append(s, x)(可能触发扩容,生成新底层数组)、s = s[1:](新 slice 结构体,可能指向不同位置)、m = map[string]int{"a": 1}(重赋值,丢弃原 map header)
  • ⚠️ 特别注意:nil slice 和 nil map 是合法值,但对它们调用 append 或写入会 panic,需先 make

想真正传引用?用指针显式控制

如果需要让函数能替换整个数据结构(比如把一个 slice 换成另一个,或把 map 清空并重建),必须传指针。这不是“Go 支持引用传递”,而是你主动传了一个指针类型的值——它依然是值传递,只是这个值恰好是指向原数据的地址。

func resetSlice(s *[]int) {
    *s = []int{0, 0, 0} // 修改指针所指的 slice 变量
}
func clearMap(m *map[string]int) {
    *m = make(map[string]int) // 替换整个 map header
}

调用时必须传地址:resetSlice(&mySlice)。漏掉 & 就只是复制了 []int 值,毫无效果。

struct 字段类型决定行为差异

结构体作为参数时,整个 struct 被复制。但字段是否“可观测修改”,取决于字段类型:

  • 字段是基础类型(intstring):形参中改它,不影响实参
  • 字段是 slice/map/*T:可通过该字段间接修改共享数据(如 s.Fields = append(s.Fields, x)Fields[]int,则底层数组可能被改)
  • 字段是 [4]int(数组):整个数组被复制,改字段内元素不影响实参

所以不要靠“struct 是否大”来决定要不要传指针,而要看你是否需要函数修改 struct 自身(如赋新值)或其内部可变容器的内容。

到这里,我们也就讲完了《Golang值传递原理与使用技巧》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!

前往漫画官网入口并下载 ➜
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>