一、什么是 range
range 是 Go 里 用于遍历数据结构的语法糖,常见用法:
Go
for k, v := range collection {
...
}
它能遍历的对象包括:
-
数组 / 切片
-
字符串
-
map
-
channel
本质上:每次循环,range 会返回"索引/键 + 值"
二、遍历数组 / 切片(最常用)
Go
arr := []int{10, 20, 30}
for i, v := range arr {
fmt.Println(i, v)
}
输出:
0 10
1 20
2 30
只要值,不要索引
Go
for _, v := range arr {
fmt.Println(v)
}
只要索引
Go
for i := range arr {
fmt.Println(i)
}
三、遍历字符串
Go
s := "你好,go!"
for i, c := range s {
fmt.Printf("index=%d,char=%c\n", i, c)
}
输出:
index=0,char=你
index=3,char=好
index=6,char=,
index=9,char=g
index=10,char=o
index=11,char=!
注意:
-
i:字节索引(不是字符索引) -
c:rune(Unicode 字符)
对比 Java
| Java | Go |
|---|---|
char 是 UTF-16 |
rune 是 UTF-32 |
String.length() 不等于字符数 |
len(s) 是字节数 |
| 遍历字符较绕 | range 天然支持 Unicode |
遍历中文,Go 用 range 是"正解"
四、遍历 map
Go
m := map[string]int{
"a": 1,
"b": 2,
}
for k, v := range m {
fmt.Println(k, v)
}
重要特性
-
map 遍历 无序
-
每次运行顺序可能不一样
-
Go 刻意这么设计(防止依赖顺序的 bug)
如果需要有序,需这样处理:
Go
keys := make([]string, 0, len(m))
for k := range m {
keys = append(keys, k)
}
sort.Strings(keys)
for _, k := range keys {
fmt.Println(k, m[k])
}
五、遍历 channel
在并发中很有用
Go
ch := make(chan int)
go func() {
ch <- 1
ch <- 2
close(ch)
}()
for v := range ch {
fmt.Println(v)
}
特点:
-
会 一直阻塞
-
直到
channel被close -
非常适合消费协程产生的数据
六、range 的"值拷贝陷阱"
问题代码
Go
arr := []int{1, 2, 3}
for _, v := range arr {
v = v * 10
}
fmt.Println(arr) // [1 2 3]
为什么没改?
-
v是 元素的拷贝 -
修改
v不影响原数组
正确写法
Go
for i := range arr {
arr[i] *= 10
}
七、什么时候不用 range?
| 场景 | 建议 |
|---|---|
| 需要修改原数组 | 用索引 for |
| 需要精确控制步长 | 用经典 for |
| 性能极限场景 | 手写 for 更可控 |
| 遍历 map 顺序敏感 | 不适合 |
八、总结
Go 的 range = 更安全、更简洁的 for-each,但要记住:
1. 值是拷贝;
2. map 无序;
3. 字符串按 rune。