值类型传参时进行值拷贝,但非深拷贝;若含引用字段如slice、map,则共享底层数组。示例中Name、Age未变,Tags被修改因指向同一数据。大对象拷贝代价高,建议使用指针传参避免性能开销。

在Go语言中,值类型(value type)在函数传参时会进行**值拷贝**,也就是说,传递的是变量的副本,而不是原始变量本身。理解这个机制对掌握Go的内存模型和性能优化非常重要。
Go中的值类型包括基本类型(如int、float64、bool)、数组、结构体(struct)等。当这些类型作为函数参数传递时,Go会创建一个该值的完整副本。这个过程是“值拷贝”,但是否算“深拷贝”取决于类型内部是否包含引用类型。
注意: 值拷贝 ≠ 深拷贝。如果结构体中包含指针、slice、map、channel等引用类型字段,这些字段指向的数据不会被复制,只是它们的引用被复制了。示例说明:
type Person struct {
Name string
Age int
Tags []string // 引用类型字段
}
func modify(p Person) {
p.Name = "Modified"
p.Age = 99
p.Tags[0] = "modified-tag"
}
func main() {
person := Person{
Name: "Alice",
Age: 25,
Tags: []string{"go", "dev"},
}
modify(person)
fmt.Println(person) // 输出:{Alice 25 [modified-tag dev]}
}
可以看到,Name 和 Age 没有被修改(因为是副本),但 Tags 被改了——因为切片是引用类型,副本中的 Tags 仍指向同一底层数组。
立即学习“go语言免费学习笔记(深入)”;
每次值传递都会在栈上分配新内存来存放副本。对于小对象(如int、小struct),开销很小,编译器还可能优化。但对于大对象(比如大数组或大结构体),频繁值拷贝会造成性能问题。
建议: 对于较大的结构体,推荐使用指针传参以避免昂贵的拷贝:
func process(p *Person) {
p.Name = "Updated"
}
Go编译器会进行逃逸分析(escape analysis),决定变量是分配在栈上还是堆上。值类型参数通常分配在栈上,函数返回后自动回收,无需GC介入,效率高。
你可以通过编译命令查看逃逸情况:
go build -gcflags="-m" your_file.go
如果看到类似“moved to heap”的提示,说明变量逃逸到了堆上,可能是因为被返回、被闭包捕获或传给了其他goroutine。
基本上就这些。理解值类型如何复制,能帮你写出更高效、更安全的Go代码。关键是:小对象无所谓拷贝,大对象尽量传指针。
以上就是Golang值类型在函数传参中如何复制_Golang value type复制与内存分配讲解的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号