Go函数参数是值传递,需传指针才能交换变量:swap(a,b int){a,b=b,a},调用时用&取地址;泛型版Swap[T any](a,b T)支持任意类型,注意避免nil指针和同一地址交换。

在 Go 语言中,函数参数默认是值传递,直接交换形参对实参无影响。要实现变量互换,必须传入变量的地址(即指针),在函数内通过解引用修改原始值。
为什么需要指针才能交换变量
Go 中所有函数参数都是副本——传入 int 就复制一个 int,传入 struct 就复制整个结构体。如果只传值,swap 函数内部交换的是两个副本,原变量完全不受影响。只有传入指针(如 *int),函数才能定位并修改原始内存位置上的数据。
基础写法:交换两个 int 变量
定义接收两个 *int 类型参数的函数,解引用后交换值:
func swap(a, b *int) {
*a, *b = *b, *a
}
立即学习“go语言免费学习笔记(深入)”;
调用时需用取地址符 & 传入变量地址:
x, y := 10, 20
swap(&x, &y)
// 此时 x == 20, y == 10
通用化:使用泛型支持任意可比较类型(Go 1.18+)
避免为每种类型重复写 swap,可用泛型封装:
func Swap[T any](a, b *T) {
*a, *b = *b, *a
}
调用方式不变,编译器自动推导类型:
name1, name2 := "Alice", "Bob"
Swap(&name1, &name2)
// name1 == "Bob", name2 == "Alice"
注意:泛型版本不要求类型可比较(comparable),因为只是赋值,不涉及比较操作。
常见错误与注意事项
- 传值而非传址:写成 swap(x, y) 会报错(类型不匹配),或误以为能生效而实际无效
- 空指针解引用:若传入 nil 指针(如 var p *int 然后调 swap(p, &x)),运行时 panic
- 交换同一变量地址:如 swap(&x, &x),虽语法合法,但无意义且易引发逻辑混淆
- 结构体字段交换:可传入字段地址,例如 swap(&s1.Name, &s2.Name),无需整个结构体指针









