Go中无内置迭代器接口,实际遍历依赖range;不同类型range返回值顺序固定,需注意零值与索引丢弃;复杂场景可用闭包封装状态,自定义类型需显式提供Iter方法或支持Range(实验性)。

Go 没有内置迭代器接口,range 就是实际使用的遍历机制
Go 语言不提供类似 Java Iterator 或 Python __iter__ 的抽象迭代器类型。所谓“迭代器模式”在 Go 中通常指手动封装状态的遍历逻辑,但绝大多数场景下,直接用 range 配合切片、map、channel 或自定义类型的方法即可满足需求。强行模拟传统迭代器(如 Next() + HasNext())反而增加复杂度,且易出错。
用 range 遍历常见集合类型时要注意返回值顺序和零值行为
range 对不同类型的返回值顺序固定:对 slice/map 返回 index/key 和 value;对 channel 只返回 value。若只关心值,应显式丢弃索引(用 _),否则编译报错或逻辑异常。
- 遍历
[]int:for i, v := range nums→i是 int 索引,v是元素副本 - 遍历
map[string]int:for k, v := range m→k是 key,v是 value 副本(不是引用) - 遍历
chan int:for v := range ch→ 仅接收值,通道关闭后自动退出 - 错误写法:
for v := range slice→ 实际得到的是索引,不是元素,极易误用
需要“可暂停/恢复”的遍历逻辑时,用闭包或结构体封装状态
当业务要求分批拉取、条件跳过、或与外部状态联动(如分页游标、IO 流读取),才考虑手写类迭代器。推荐用函数返回闭包,比 struct + 方法更轻量。
func NewIntIterator(nums []int) func() (int, bool) {
i := 0
return func() (int, bool) {
if i >= len(nums) {
return 0, false
}
val := nums[i]
i++
return val, true
}
}
// 使用:
next := NewIntIterator([]int{10, 20, 30})
for {
if val, ok := next(); !ok {
break
}
fmt.Println(val)
}
注意:闭包捕获的是 nums 的引用,若原切片被修改,迭代结果可能变化;若需稳定快照,应在闭包内做 copy。
立即学习“go语言免费学习笔记(深入)”;
自定义类型实现 range 支持需满足特定方法签名
Go 允许为自定义类型添加 Range 方法(非接口强制),使 range 可用于该类型 —— 这是实验性特性(Go 1.23+),尚未稳定。当前稳定方式仍是暴露一个可 range 的字段(如 Items []T)或提供 Iter() func() (T, bool) 方法。
- 不要试图实现
Iterator接口并期望range自动识别 —— Go 没这机制 - 避免在结构体中暴露未加锁的可变切片供外部
range,并发读写会 panic - 若类型底层是 map,
range顺序不确定,勿依赖遍历序
range 一行解决。所谓“迭代器模式”,在 Go 中更多是意识:把遍历逻辑从消费代码中解耦出来,而不是套用 OOP 的类继承模板。状态管理、边界检查、资源释放这些细节,得自己盯住。










