登录
首页 >  Golang >  Go教程

Golang函数传参是值还是引用?

时间:2026-04-13 21:43:05 474浏览 收藏

Go语言中函数参数永远是传值,所谓“传引用”只是slice、map、chan等类型因底层包含指针字段而产生的表象;array则是彻底的值传递,修改不影响原值;使用指针参数并非为了“能修改”,而是出于明确的语义需求(如修改原数据)、性能考虑(避免大对象拷贝)或接口约束(指针接收者方法),且Go强制显式取地址,让内存共享关系清晰可见——理解这一点,才能写出既高效又安全的Go代码。

Golang函数参数传值还是传引用_Golang函数参数传递教程【总结】

Go 语言中函数参数**永远是传值(pass by value)**,不存在传统意义上的“传引用”。所谓“传指针”,本质仍是传指针变量的副本——只是这个副本恰好能间接修改原数据。

为什么 slicemapchan 看起来像“传引用”?

因为它们底层是结构体,包含指向底层数据的指针字段。传参时虽复制结构体,但其中的指针字段仍指向同一块内存。

  • slice 是三字段结构体:ptr(指向底层数组)、lencap;修改元素会生效,但 append 后若扩容,新 slice 的 ptr 可能已变,原调用方看不到变化
  • mapchan 同理:内部含指针或句柄,所以增删元素可见,但重新赋值(如 m = make(map[string]int))不影响外部变量
  • 反例:array(如 [3]int)是纯值类型,传参会完整复制,修改不影响原数组

func f(x *int)func f(x int) 的关键区别

前者接收一个指针类型的值,后者接收一个整型值。区别不在“是否可修改”,而在“能否让修改反映到调用方变量上”。

  • *int:函数内解引用后赋值(如 *x = 42),调用方原始变量值会变
  • int:函数内改 x 本身,只影响副本,调用方无感知
  • 注意:传 *int 并不意味着安全——如果传入的是 nil 指针,解引用会 panic:panic: runtime error: invalid memory address or nil pointer dereference

什么时候该用指针参数?看三个实际信号

不是“想改就传指针”,而是基于语义和效率做判断。

  • 需要修改调用方变量所指向的数据(如更新结构体字段、重置状态)→ 用 *T
  • 值类型过大(如大 struct),避免拷贝开销 → 用 *T 提升性能(但需权衡可读性与逃逸分析影响)
  • 类型实现了某个接口,且方法集要求指针接收者(如 sync.Mutex.Lock() 必须用 *sync.Mutex 调用)→ 参数也得是 *T,否则无法满足接口

最易被忽略的一点:Go 没有隐式取地址。你传 myStructfunc f(*MyStruct) 会编译失败,必须显式写 f(&myStruct)。这既是约束,也是明确性保障——谁在共享内存,一眼可知。

终于介绍完啦!小伙伴们,这篇关于《Golang函数传参是值还是引用?》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>