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

在Go语言中,使用
for range
循环遍历一个指针切片时,最需要注意的是循环变量的重用问题,尤其是在取地址或启动协程时容易引发意外行为。
循环变量会被复用
Go的
for range
循环中,
range
使用的循环变量在每次迭代中会被**复用**,而不是重新声明。这意味着如果你在循环中取变量的地址,所有地址可能指向同一个内存位置。
例如:
ptrs := []*int{{1}, {2}, {3}}var addrs []*intfor _, 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
就能避免大部分陷阱。
以上就是Golang中for range循环一个指针切片时需要注意什么问题的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1402366.html
微信扫一扫
支付宝扫一扫