range遍历时修改v不生效,因v是副本;需用slice[i]修改;goroutine中复用v需显式复制或传参;遍历时不可增删切片元素,应先收集索引再批量处理。
range 遍历切片时修改元素值不生效直接对 range 返回的值变量赋值,不会影响原切片内容,因为 Go 中 range 的第二个返回值是元素的副本,不是引用。
for i, v := range slice {
v = v * 2 // 这里改的是 v 的副本,slice[i] 不变
}slice[i] = slice[i] * 2 或 slice[i] *= 2
v 更安全;如果要写入,必须用 slice[i]
range 循环中意外复用变量地址在循环内启动 goroutine 并传入 v 或 &v 是常见陷阱:所有 goroutine 最终可能看到同一个 v 的最终值,因为 v 在每次迭代中被复用。
for _, v := range data {
go func() {
fmt.Println(v) // 所有 goroutine 都打印最后一个 v
}()
}for _, v := range data {
v := v // 创建新变量,遮蔽外层 v
go func() {
fmt.Println(v)
}()
}for _, v := range data {
go func(val int) {
fmt.Println(val)
}(v)
}不能。在 range 迭代过程中直接对底层数组扩容(如 append)可能导致部分元素被跳过,甚至引发不可预测行为,因为 range 在开始时已确定迭代长度和底层数组指针。
range 迭代仍按原始长度进行,新增元素不会被访问到slice = append(slice[:i], slice[i+1:]...) 会改变底层数组结构,但 range 不感知for i := 0; i 并手动控制 i 增减
range 还是传统 for 索引循环两者编译后生成的汇编几乎一致,性能无实质差别。选择依据应是语义清晰度与安全性,而非微小性能差异。
range 更适合只读遍历、无需索引的场景,代码简洁且不易越界for i := 0; i 更直接
len(slice) 在循环条件中每次都会求值,但它是 O(1) 操作,无负担;若切片长度在循环中可能变化且你依赖它,才需提前缓存实际写代码时,最容易被忽略的是:range 的第二个变量永远是值拷贝,哪怕你对它取地址,得到的也是那个临时变量的地址,不是原切片元素的地址。这点在调试并发逻辑或结构体字段更新时,特别容易翻车。