go 语言通过闭包模拟迭代器:工厂函数接收集合,内部捕获索引变量并返回无参 next 函数,每次调用返回元素及是否结束标志,隐藏遍历状态,符合迭代器原则。

在 Go 语言中没有内置的迭代器接口(如 java 的 Iterator 或 python 的 __iter__),但可以通过函数式风格和闭包轻松模拟迭代器模式,实现对集合的安全、按需遍历。
用闭包封装状态,返回 next 函数
最自然的 Go 风格是返回一个无参函数,每次调用返回下一个元素和是否结束的标志。这种设计隐藏了内部索引或游标,符合迭代器“只暴露遍历能力,不暴露数据结构”的原则。
- 定义一个工厂函数,接收集合(如切片、map、自定义结构体)作为参数
- 在函数体内捕获索引变量(如
i := -1),并返回闭包func() (T, bool) - 每次调用该函数时递增索引,检查边界,返回对应元素和
true;越界则返回零值和false
例如遍历字符串切片:
func NewStringiterator(ss []String) func() (string, bool) {
i := -1
return func() (string, bool) {
i++
if i return ss[i], true
}
return “”, false
}
}
为自定义容器实现迭代器方法
如果你有一个结构体表示集合(比如链表、树、缓存等),可在其上定义 Iterator() 方法,返回统一的迭代函数类型,增强可组合性。
立即学习“go语言免费学习笔记(深入)”;
- 推荐使用类型别名统一签名:
type Iterator[T any] func() (T, bool) -
Iterator()方法内部构造并返回闭包,访问结构体字段(如 head 指针、items 切片) - 这样用户无需关心底层结构,只与迭代行为交互,便于测试和替换实现
支持中断与复用:避免一次性消费
Go 中的闭包迭代器默认是“单次可遍历”的(类似生成器),但如果需要多次遍历,不要在工厂函数里直接修改原始状态;而是每次调用 Iterator() 都新建一个独立闭包。
- 错误做法:把索引存在结构体字段里,导致并发或重复调用出错
- 正确做法:状态完全封闭在闭包内,每次
Iterator()调用都初始化新状态 - 若需暂停/恢复(如处理中间结果后继续),直接保存当前迭代器函数变量即可
结合 for-range 简化使用
虽然 Go 没有原生支持,但你可以让迭代器配合标准 for 循环,写法接近惯用 style:
it := NewStringIterator([]string{“a”, “b”, “c”})
for s, ok := it(); ok; s, ok = it() {
fmt.Println(s)
}
也可以封装成更简洁的辅助函数(如 foreach),接受迭代器和处理函数,内部完成循环逻辑。
基本上就这些。Go 的迭代器不是靠接口强制,而是靠函数值 + 闭包 + 类型约束(Go 1.18+)来达成解耦与复用。不复杂但容易忽略状态隔离和泛型适配。