Golang的参数传递机制是很多开发者入门时会困惑的问题,网络上也存在不同的说法,有人认为是值传递,有人觉得部分场景是引用传递。实际上从官方定义和底层实现来看,Golang所有的参数传递本质都是值传递,不存在引用传递的情况。
Golang参数传递的核心定义
值传递指的是在调用函数时,会将实参的副本传递给函数的形参,函数内部对形参的修改不会影响到原来的实参。而引用传递是指传递的是实参的引用地址,函数内部修改形参会直接改变原实参的内容。Golang中不管是基本类型、结构体还是切片、map等引用类型,传递时都是传递的实参的副本,因此属于值传递。
不同参数类型的传递表现
基本类型的传递
基本类型包括int、string、bool、float等,传递时直接复制值,函数内的修改不会影响原变量。
package main
import "fmt"
// 定义修改int值的函数
func modifyInt(num int) {
num = 100
fmt.Println("函数内num的值:", num)
}
func main() {
a := 10
modifyInt(a)
// 原变量a的值不会被修改
fmt.Println("函数外a的值:", a)
}
上述代码执行后,函数内输出的num是100,函数外输出的a仍然是10,符合值传递的特征。
结构体类型的传递
结构体也是值类型,传递时会复制整个结构体的内容,修改形参的字段不会影响原结构体。
package main
import "fmt"
// 定义用户结构体
type User struct {
Name string
Age int
}
func modifyUser(u User) {
u.Name = "李四"
u.Age = 20
fmt.Println("函数内u的信息:", u)
}
func main() {
user := User{Name: "张三", Age: 18}
modifyUser(user)
fmt.Println("函数外user的信息:", user)
}
执行后可以看到,函数外user的信息仍然是张三18岁,没有被修改。
切片、map等类型的传递
切片和map的底层结构包含一个指向底层数据的指针,传递时复制的是这个切片或map的结构体副本,两个副本的指针指向同一个底层数据,因此修改副本的元素内容会影响原变量,这也正是很多人误以为这是引用传递的原因。
package main
import "fmt"
func modifySlice(s []int) {
// 修改切片的元素,会影响原切片
s[0] = 100
// 给切片追加元素,不会改变原切片的长度和容量
s = append(s, 4)
fmt.Println("函数内切片:", s)
}
func main() {
sl := []int{1, 2, 3}
modifySlice(sl)
fmt.Println("函数外切片:", sl)
}
执行后可以看到,原切片的第一个元素被修改成了100,但是追加的元素4没有出现在原切片中,因为函数内append后形参指向了新的底层数组,而原实参的指针没有变化,这也说明传递的是副本,不是引用。
如何区分值传递和引用传递
判断是值传递还是引用传递的核心标准是:函数内对形参的重新赋值是否会影响原实参。如果是值传递,重新赋值形参不会影响原实参;如果是引用传递,重新赋值形参会直接改变原实参。
用切片的例子验证,在上面的modifySlice函数中,我们将s重新赋值为append后的结果,但是原切片sl并没有变化,说明传递的是副本,属于值传递。
总结
Golang中所有的参数传递都是值传递,不存在引用传递。对于基本类型和结构体,传递的是值的完整副本,修改不会影响原变量;对于切片、map、channel等类型,传递的是其底层结构体的副本,副本中的指针指向原底层数据,因此修改元素内容会影响原变量,但重新赋值形参不会影响原实参。理解这个核心逻辑,就能避免开发中出现参数传递相关的错误。