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 的“副本语义”和各类型底层行为,写遍历时就能少踩坑、更安心。