最需要注意循环变量被复用的问题,导致取地址或goroutine中捕获的值异常;应在每次迭代时创建局部副本避免该问题。

在Go语言中,使用
for range循环遍历一个指针切片时,最需要注意的是循环变量的重用问题,尤其是在取地址或启动协程时容易引发意外行为。
循环变量会被复用
Go的
for range循环中,
range使用的循环变量在每次迭代中会被**复用**,而不是重新声明。这意味着如果你在循环中取变量的地址,所有地址可能指向同一个内存位置。
例如:
ptrs := []*int{{1}, {2}, {3}}
var addrs []*int
for _, p := range ptrs {
addrs = append(addrs, &p) // 错误:&p 始终指向同一个变量
}
// 此时 addrs 中所有元素的地址都相同,值也相同(可能是最后一个值)
上面代码中,
p是每次迭代被赋值的变量,但它在整个循环中是同一个变量。因此
&p取的是同一个地址,最终
addrs中所有指针都指向同一个值,结果不可预期。
立即学习“go语言免费学习笔记(深入)”;
在goroutine中捕获循环变量要小心
当你在
for range中启动多个goroutine,并传入循环变量时,如果不注意,所有goroutine可能看到的是同一个值。
for _, p := range ptrs {
go func() {
fmt.Println(*p) // 可能全部打印相同的值
}()
}
这是因为
p在每次迭代中被修改,而所有闭包都引用了同一个变量。结果是输出可能全部一样,甚至出现数据竞争。
解决方法是:
- 在循环内部创建局部副本:
for _, p := range ptrs {
p := p // 创建新的变量 p,作用域在本次迭代
go func() {
fmt.Println(*p)
}()
}
对指针切片本身的操作要小心
如果你在遍历指针切片时修改了切片本身(如追加、删除),可能导致意外结果,因为
range在开始时就决定了遍历的长度和元素。
例如:
for i, p := range ptrs {
if i == 0 {
ptrs = append(ptrs, &newVal)
}
// 仍然只遍历原始长度的元素,新增的不会被遍历到
}
这是
range的设计行为,不是指针特有的问题,但结合指针使用时容易忽略。
基本上就这些。关键是理解
for range中变量的复用机制,避免取地址或在闭包中直接使用循环变量。简单加一行
p := p就能避免大部分陷阱。










