for range遍历切片/数组得用索引修改原数据,map遍历值为副本且地址复用,字符串按rune遍历并返回字节偏移,channel遍历持续接收至关闭。

Go 语言中 for range 是遍历集合最常用、最简洁的方式,它能自动解构元素(甚至索引),但不同数据类型的遍历行为有关键差异——理解这些差异,才能避免常见坑,比如修改切片元素不生效、遍历 map 时值被复用等。
切片/数组:拿到的是元素副本,想改原数据得用索引
遍历时,range 默认返回的是当前元素的**副本**,直接赋值不会影响原切片:
nums := []int{1, 2, 3}
for _, v := range nums {
v = v * 2 // 这里改的是 v 的副本,nums 不变
}
// nums 还是 [1 2 3]
如果要修改原切片,必须通过索引访问:
- 只用索引:
for i := range nums { nums[i] *= 2 } - 同时要索引和值:
for i, v := range nums { nums[i] = v * 2 }
map:遍历顺序不固定,value 是副本,地址可能复用
Go 中 map 遍历无序(每次运行结果可能不同),且 v 是 value 的副本:
立即学习“go语言免费学习笔记(深入)”;
m := map[string]int{"a": 1, "b": 2}
for k, v := range m {
v++ // 不影响 m[k]
}
更隐蔽的坑是取地址:
var ptrs []*int
for _, v := range m {
ptrs = append(ptrs, &v) // 所有指针都指向同一个内存地址!
}
解决方法:用局部变量保存再取地址,或直接用索引(如 &m[k])。
字符串:range 按 rune 遍历,不是 byte
中文、emoji 等 Unicode 字符在 UTF-8 中占多个字节,但 for range 自动按 Unicode 码点(rune)拆分:
s := "Go❤️"
for i, r := range s {
fmt.Printf("index %d: rune %U (%c)\n", i, r, r)
}
// 输出:
// index 0: U+0047 (G)
// index 2: U+006F (o)
// index 4: U+2764 (❤)
注意:i 是字节偏移位置,不是 rune 序号;要获取第 n 个字符,建议先转为 []rune(s) 再索引。
channel:range 会持续接收直到关闭
对 channel 使用 for range,等价于不断 ,直到 channel 被关闭:
ch := make(chan int, 2)
ch <- 1
ch <- 2
close(ch)
for v := range ch { // 自动退出,不会阻塞
fmt.Println(v)
}
注意:未关闭的 channel 会永久阻塞;已关闭但仍有值未读完,range 会读完再退出。
基本上就这些。掌握 for range 的“副本语义”和各类型底层行为,写遍历时就能少踩坑、更安心。









