本文永久链接 – https://tonybai.com/2025/05/29/xiter-declined
大家好,我是Tony Bai。
随着 Go 1.22 中 range over func 实验性特性的引入,以及在 Go 1.23 中该特性的最终落地(#61405),Go 社区对迭代器(Iterators)的讨论达到了新的高度。在这一背景下,一项旨在提供标准迭代器适配器(Adapters)的提案 x/exp/xiter (Issue #61898) 应运而生,曾被寄予厚望,期望能为 Go 开发者带来一套便捷、统一的迭代器操作工具集。然而,经过社区的广泛讨论和官方团队的审慎评估,该提案最终被标记为“婉拒并撤回 (declined as retracted)”。本文将对 x/exp/xiter 提案的核心内容做个简单解读,说说社区围绕它的主要争论点,以及最终导致其搁浅的关键因素,并简单谈谈这一决策对 Go 语言生态的潜在影响与启示。
x/exp/xiter 提案由 Russ Cox (rsc) 发起,旨在 golang.org/x/exp/xiter 包中定义一系列迭代器适配器。这些适配器主要服务于 Go 1.23 中引入的 range over func 特性,提供诸如数据转换 (Map)、过滤 (Filter)、聚合 (Reduce)、连接 (Concat)、并行处理 (Zip) 等常用功能。
其核心目标是:
提案中包含了一系列具体的函数定义,例如:
值得注意的是,许多函数都提供了针对 iter.Seq[V](单值序列)和 iter.Seq2[K, V](键值对序列)的两个版本,这导致了 API 数量上的成倍增加。
以下是一个简单的设想用法示例:
package main
import (
"fmt"
"iter"
// 假设 xiter 包已存在且包含提案中的函数
// "golang.org/x/exp/xiter"
)
// 假设的 Filter 函数
func Filter[V any](f func(V) bool, seq iter.Seq[V]) iter.Seq[V] {
return func(yield func(V) bool) {
for v := range seq {
if f(v) && !yield(v) {
return
}
}
}
}
// 假设的 Map 函数
func Map[In, Out any](f func(In) Out, seq iter.Seq[In]) iter.Seq[Out] {
return func(yield func(Out) bool) {
for in := range seq {
if !yield(f(in)) {
return
}
}
}
}
func main() {
numbers := func(yield func(int) bool) {
for i := 1; i <= 5; i++ {
if !yield(i) {
return
}
}
}
// 设想:筛选偶数,然后平方
evenSquares := Map(
func(n int) int { return n * n },
Filter(
func(n int) bool { return n%2 == 0 },
numbers,
),
)
for sq := range evenSquares {
fmt.Println(sq) // 预期输出: 4, 16
}
}
x/exp/xiter 提案引发了社区成员的广泛讨论,焦点集中在 API 设计、易用性、与 Go 语言既有哲学的契合度等多个方面。
提案中大量函数针对 iter.Seq[V] 和 iter.Seq2[K, V] 提供了两个版本(例如 Map 和 Map2),这直接导致了 API 接口数量的翻倍。虽然 Russ Cox 认为这只是“重复而非复杂性”,因为学习了 Foo 形式后,Foo2 形式只是一个简单的规则,但仍有社区成员担忧这会使包显得臃肿,影响开发者体验,并随着未来可能增加更多适配器而使问题恶化。
提案中的 Zip 函数设计为当一个序列耗尽后,仍会继续迭代另一个序列,并在 Zipped 结构体中通过 Ok1/Ok2 标志位标示元素是否存在。这与 Python 等语言中 zip 在最短序列结束时即停止的行为不同,更类似于 zip_longest。社区开发者就此展开讨论,认为应提供传统意义上的 Zip(返回 Seq2[V1, V2] 并在短序列结束时停止)和行为类似 zip_longest 的版本(如 ZipAll 或将提案中的 Zip 重命名为 ZipLongest)。
在 Go 1.23 发布一段时间后,经过充分的讨论和实践反馈,Russ Cox 和 Austin Clements 代表提案审查小组,宣布将此提案标记为“婉拒并撤回 (declined as retracted)”。
主要原因可以归纳为:
x/exp/xiter 提案的搁浅,并不意味着 Go 语言在迭代器支持上的停滞。相反,它反映了 Go 团队在语言发展上一贯的审慎和务实态度。
对 Go 开发者而言,这意味着:
x/exp/xiter 的讨论过程本身就是一次宝贵的社区实践,它汇集了众多 Go 开发者的智慧与经验,即便提案未被接纳,其间的深入思考和论证也为 Go 语言迭代器生态的未来发展指明了方向,并留下了丰富的参考。我们期待看到 Go 社区在迭代器领域持续探索,涌现出更多符合 Go 风格且能切实解决开发者痛点的优秀工具与实践。
商务合作方式:撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。如有需求,请扫描下方公众号二维码,与我私信联系。
© 2025, bigwhite. 版权所有.