Go 中有 foreach 循环吗?

Is there a foreach loop in Go?

提问人:tatsuhirosatou 提问时间:10/16/2011 最后编辑:Tiago Peczenyjtatsuhirosatou 更新时间:11/10/2022 访问量:585804

问:

Go 语言中有结构吗? 我可以使用 a 遍历切片或数组吗?foreachfor

foreach 切片

评论

1赞 Kakashi 10/16/2011
退房: groups.google.com/group/golang-nuts/browse_thread/thread/...
2赞 kostix 10/16/2011
在 Go 教程的“关于类型的插曲”部分(接近尾声)中也提到了 in 循环的用法。rangefor

答:

1115赞 6 revs, 5 users 31%davetron5000 #1

带有 range 子句的 From For 语句

带有“range”子句的“for”语句遍历所有条目 数组、切片、字符串或映射,或通道上接收的值。 对于每个条目,它将迭代值分配给相应的迭代 变量,然后执行该块。

举个例子:

for index, element := range someSlice {
    // index is the index where we are
    // element is the element from someSlice for where we are
}

如果你不关心索引,你可以使用:_

for _, element := range someSlice {
    // element is the element from someSlice for where we are
}

下划线 是空白标识符,是一个匿名占位符。_

评论

27赞 Brent Bradburn 12/9/2018
在此示例中,是元素(副本)的 -- 它不是元素本身。尽管您可以分配给 ,但这不会影响基础序列。elementelement
0赞 Sergiy Kolodyazhnyy 12/29/2018
我知道在 Python 和 C 中经常使用下划线作为本地化的函数(即 gettext)。使用下划线会导致 Go 出现任何问题吗?Go 甚至使用相同的库进行本地化吗?
4赞 Davos 4/5/2019
@SergiyKolodyazhnyy Py 文档说“(gettext) 函数通常别名为本地命名空间”,这只是按照惯例,它不是本地化库的一部分。下划线是一个有效的标签,它也是 Go(以及 Python、Scala 和其他语言)中的约定,可以分配给您不会使用的返回值。此示例中的范围仅限于循环的主体。如果您有一个包范围的函数,那么它将在 for 循环的作用域内隐藏。有一些用于本地化的包,我没有看到任何用作函数名称的软件包。_()___for__
0赞 kapad 9/8/2020
请参阅下面的 Moshe Revah 的回答,了解 的更多使用示例。包括切片、地图和频道。for...range
13赞 ceving 1/30/2014 #2

下面的示例演示如何在循环中使用运算符来实现循环。rangeforforeach

func PrintXml (out io.Writer, value interface{}) error {
    var data []byte
    var err error

    for _, action := range []func() {
        func () { data, err = xml.MarshalIndent(value, "", "  ") },
        func () { _, err = out.Write([]byte(xml.Header)) },
        func () { _, err = out.Write(data) },
        func () { _, err = out.Write([]byte("\n")) }} {
        action();
        if err != nil {
            return err
        }
    }
    return nil;
}

该示例循环访问函数数组,以统一函数的错误处理。一个完整的例子是在谷歌的游乐场

PS:它还表明,悬挂大括号对于代码的可读性来说是一个坏主意。提示:条件在通话前结束。很明显,不是吗?foraction()

评论

3赞 topskip 7/7/2014
添加一个,条件结束的地方就更清楚了:play.golang.org/p/pcRg6WdxBd - 这实际上是我第一次找到对这种风格的反驳,谢谢!,forgo fmt
0赞 Filip Haglund 6/14/2015
@topskip两者都是有效的;只需选择最好的:)
0赞 topskip 6/14/2015
@FilipHaglund 如果它有效,那不是重点。关键是 IMO 在上面的特定情况下,for 条件在哪里结束更清楚。
12赞 AndreasHassing 2/3/2016
在我看来,对于目标问题来说,这个答案过于复杂。
0赞 ceving 1/16/2019
@AndreasHassing 如何在不引入冗余的情况下做到这一点?
227赞 Zippo 12/17/2015 #3

Go 具有类似 -的语法。它支持数组/切片、映射和通道。foreach

遍历数组切片

// index and value
for i, v := range slice {}

// index only
for i := range slice {}

// value only
for _, v := range slice {}

遍历地图

// key and value
for key, value := range theMap {}

// key only
for key := range theMap {}

// value only
for _, value := range theMap {}

通道上迭代:

for v := range theChan {}

遍历一个通道等同于从通道接收,直到它被关闭:

for {
    v, ok := <-theChan
    if !ok {
        break
    }
}

评论

22赞 domoarigato 8/2/2016
虽然 OP 只要求切片使用,但我更喜欢这个答案,因为大多数人最终也需要其他用法。
4赞 colm.anseo 10/11/2017
关于用法的重要区别:如果写入器在某个时候关闭通道,则在通道上扩展将优雅地退出循环。在等效的情况下,它不会在通道关闭时退出。您可以通过第二个返回值对此进行测试。旅游示例chanfor {v := <-theChan}ok
0赞 Levite 12/22/2017
阅读时也这么想,代表无限循环。for { ... }
0赞 Duong Phan 3/22/2022
没有两个键,值怎么样。我只想运行特定的 len。并且不对数组执行任何操作。例如:for _ := range slice{}
12赞 robstarbuck 9/6/2016 #4

实际上,您可以通过对 type 使用 来不引用其返回值:rangefor range

arr := make([]uint8, 5)
i,j := 0,0
for range arr {
    fmt.Println("Array Loop", i)
    i++
}

for range "bytes" {
    fmt.Println("String Loop", j)
    j++
}

https://play.golang.org/p/XHrHLbJMEd

评论

3赞 Sridhar 9/6/2016
很高兴知道,但在大多数情况下,这不会有用
0赞 robstarbuck 9/6/2016
同意@Sridhar这是相当小众的。
24赞 Nisal Edu 10/3/2017 #5

以下是如何在 Go 中使用 foreach 的示例代码:

package main

import (
    "fmt"
)

func main() {

    arrayOne := [3]string{"Apple", "Mango", "Banana"}

    for index,element := range arrayOne{

        fmt.Println(index)
        fmt.Println(element)

    }

}

这是一个运行示例 https://play.golang.org/p/LXptmH4X_0

评论

2赞 Gwyneth Llewelyn 6/12/2020
有时,最简单的例子是最有用的。谢谢!我不反对其他评论者最深奥的答案——他们当然说明了非常惯用的 Go 编程的复杂性,以至于它们变成了......不可读且难以理解——但我更喜欢你的答案:它以最简单的示例直击核心(这有效,而且很明显为什么有效)。
1赞 jpihl 5/25/2018 #6

这可能是显而易见的,但您可以像这样内联数组:

package main

import (
    "fmt"
)

func main() {
    for _, element := range [3]string{"a", "b", "c"} {
        fmt.Print(element)
    }
}

输出:

abc

https://play.golang.org/p/gkKgF3y5nmt

16赞 Amitesh Bharti 8/15/2018 #7

是的,范围

for 循环的范围形式循环访问切片或映射。

在切片上取范围时,每次迭代返回两个值。第一个是索引,第二个是该索引处的元素的副本。

例:

package main

import "fmt"

var pow = []int{1, 2, 4, 8, 16, 32, 64, 128}

func main() {
    for i, v := range pow {
        fmt.Printf("2**%d = %d\n", i, v)
    }

    for i := range pow {
        pow[i] = 1 << uint(i) // == 2**i
    }
    for _, value := range pow {
        fmt.Printf("%d\n", value)
    }
}
  • 您可以通过将 分配给 来跳过索引或值。_
  • 如果只需要索引,请完全删除该值。,
0赞 Jose78 8/24/2020 #8

我刚刚实现了这个库:https://github.com/jose78/go-collection

以下是如何使用 Foreach 循环的示例:

package main

import (
    "fmt"

    col "github.com/jose78/go-collection/collections"
)

type user struct {
    name string
    age  int
    id   int
}

func main() {
    newList := col.ListType{user{"Alvaro", 6, 1}, user{"Sofia", 3, 2}}
    newList = append(newList, user{"Mon", 0, 3})

    newList.Foreach(simpleLoop)

    if err := newList.Foreach(simpleLoopWithError); err != nil{
        fmt.Printf("This error >>> %v <<< was produced", err )
    }
}

var simpleLoop col.FnForeachList = func(mapper interface{}, index int) {
    fmt.Printf("%d.- item:%v\n", index, mapper)
}


var simpleLoopWithError col.FnForeachList = func(mapper interface{}, index int) {
    if index > 1{
        panic(fmt.Sprintf("Error produced with index == %d\n", index))
    }
    fmt.Printf("%d.- item:%v\n", index, mapper)
}

此执行的结果应为:

0.- item:{Alvaro 6 1}
1.- item:{Sofia 3 2}
2.- item:{Mon 0 3}
0.- item:{Alvaro 6 1}
1.- item:{Sofia 3 2}
Recovered in f Error produced with index == 2

ERROR: Error produced with index == 2
This error >>> Error produced with index == 2
 <<< was produced

在 playGrounD 中尝试此代码

评论

0赞 Peter Mortensen 5/23/2022
它如何成为 for-each 循环?不就是列表处理吗?你能详细说明你的答案吗?(但没有“编辑:”、“更新:”或类似内容 - 问题/答案应该看起来就像今天写的一样。
1赞 Blue Leo 11/10/2022 #9

我看到很多使用范围的例子。只需提醒一下该范围,即可创建您正在迭代的任何内容的副本。如果对 foreach 范围内的内容进行更改,则不会更改原始容器中的值,在这种情况下,您将需要一个传统的 for 循环,其中包含递增的索引和 deference 索引引用。例如:

for i := 0; i < len(arr); i++ {
    element := &arr[i]
    element.Val = newVal
}