在Go语言中有foreach结构吗? 我可以迭代一个片或数组使用for?
当前回答
下面的例子展示了如何在for循环中使用range操作符来实现foreach循环。
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:这也表明挂大括号对于代码的可读性来说是一个坏主意。提示:for条件刚好在action()调用之前结束。很明显,不是吗?
其他回答
我看到很多使用范围的例子。只要注意一下,这个范围就会创建一个你迭代的对象的副本。如果对foreach范围内的内容进行更改,则不会更改原始容器中的值,在这种情况下,您将需要一个传统的for循环,其中包含一个递增的索引和遵循索引引用。例如:
for i := 0; i < len(arr); i++ {
element := &arr[i]
element.Val = newVal
}
下面是如何在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
下面的例子展示了如何在for循环中使用range操作符来实现foreach循环。
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:这也表明挂大括号对于代码的可读性来说是一个坏主意。提示:for条件刚好在action()调用之前结束。很明显,不是吗?
这可能是显而易见的,但你可以像这样内联数组:
package main
import (
"fmt"
)
func main() {
for _, element := range [3]string{"a", "b", "c"} {
fmt.Print(element)
}
}
输出:
abc
https://play.golang.org/p/gkKgF3y5nmt
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
}
}