Go函数参数始终值传递,即传递数据副本。基本类型修改不影响原值;传指针时地址副本指向同一内存,可修改原内容;slice、map等引用类型传递结构体副本,但内部指针仍指向原数据,故修改元素有效,扩容则不影响原变量;大结构体建议传指针以避免开销。
Golang 中的函数参数传递始终是值传递,也就是说,函数接收到的是原始数据的一个副本,而不是原始数据本身。理解这一点对掌握 Go 的行为逻辑至关重要,尤其是在处理结构体、指针、切片、map 等类型时。
所谓值传递,是指在调用函数时,实参的值被复制一份,然后传递给函数的形参。函数内部对参数的修改只作用于副本,不会影响原始变量。
例如:
func modify(a int) {这里 a 是 x 的副本,modify 函数中对 a 的修改不影响 x。
立即学习“go语言免费学习笔记(深入)”;
有人误以为传指针就是“引用传递”,其实不然。Go 仍然是值传递,只不过这个“值”是一个地址。
看下面的例子:
func modifyByPtr(p *int) {这里传递的是 &x,即 x 的地址。函数接收的是这个地址的副本,但副本指向的仍是同一块内存。因此通过 *p 修改内容,会影响原始变量。
关键点:指针的值(地址)被复制,但指向的内存是同一块。
Go 中的 slice、map、channel 都是引用类型,但它们的底层结构仍然是值传递。
以 slice 为例:
func appendToSlice(s []int) {虽然 slice 内部包含指向底层数组的指针,但传递时,slice 的结构体(包含指针、长度、容量)被整体复制。append 操作可能导致扩容,从而让副本指向新数组,不影响原 slice。
但如果只是修改元素:
func changeElement(s []int) {因为副本中的指针仍指向原底层数组,修改元素会反映到原 slice。
结构体在传递时会被完整复制,如果结构体较大,值传递会带来性能开销。
推荐做法:传递结构体指针以避免复制。
type User struct {这样既避免了复制大对象,又能修改原结构体。
基本上就这些。Go 的值传递机制统一而清晰:所有参数都复制一份传入。所谓“引用类型能被修改”,是因为它们内部包含指针,副本依然指向同一块数据。理解这一点,就能准确预测函数调用的行为。
以上就是怎样理解Golang的值传递特性 分析函数参数传递的底层机制的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 //m.sbmmt.com/ All Rights Reserved | php.cn | 湘ICP备2023035733号