Go语言中参数传递只有值传递,传递的是数据副本或指针副本;基本类型修改不影响原值,指针、slice、map、channel因副本指向同一地址可修改原始数据。

Go语言中的参数传递看似简单,但理解其底层机制对编写高效、安全的代码至关重要。很多人误以为Go存在“值传递”和“引用传递”的区分,实际上Go函数传参只有值传递。无论是基本类型、指针、slice、map还是channel,传入函数的都是副本。关键在于:副本的内容是什么?是指向堆内存的指针?还是整个数据结构本身?这决定了我们能否在函数内修改原始数据。
在Go中,每次调用函数时,实参会复制一份到被调函数的栈空间中。这个复制过程就是值传递的核心。比如:
func modifyValue(x int) { x = 100 }func main() {
a := 10
modifyValue(a)
fmt.Println(a) // 输出 10,原值未变
}
这里 x 是 a 的副本,修改 x 不影响 a。同理,如果传的是指针:
func modifyPointer(p *int) { *p = 100 }func main() {
a := 10
modifyPointer(&a)
fmt.Println(a) // 输出 100
}
虽然仍然是值传递——传递的是指针的副本,但由于副本和原指针指向同一地址,通过解引用可以修改原始数据。这就是为什么指针能“改变外部变量”的原因。
立即学习“go语言免费学习笔记(深入)”;
这些类型常让人困惑,因为它们在函数内修改元素时,外部也会看到变化。但这并不意味着它们是“引用传递”。真实原因是:这些类型的底层结构包含指向数据的指针。
例如:
func modifySlice(s []int) { s[0] = 999 }func main() {
arr := []int{1, 2, 3}
modifySlice(arr)
fmt.Println(arr) // [999 2 3]
}
尽管是值传递,但副本 slice 仍指向原底层数组,所以修改生效。
每次函数调用,Go会在栈上分配一个栈帧(stack frame),用于存放参数、局部变量和返回地址。参数从调用者栈帧复制到被调函数栈帧中。当函数返回,栈帧销毁,所有局部变量和参数随之释放。
以64位系统为例,一个 int 占8字节,指针也占8字节。传大结构体时,复制开销大,建议传指针避免性能问题:
type LargeStruct struct { data [1000]int }func processByValue(ls LargeStruct) { / 复制整个结构体 / }
func processByPointer(ls LargeStruct) { / 只复制指针 */ }
后者更高效,因为只复制8字节指针,而非8KB数据。
基本上就这些。Go的参数传递统一而清晰:永远是值传递。区别只在于传的是数据本身,还是指向数据的指针。理解这一点,结合内存布局和栈帧机制,就能准确预判函数调用对数据的影响。
以上就是如何理解Golang参数传递的本质_Golang内存地址与栈帧结构解析的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号