Czy w Go jest pętla Foreach?

565

Czy foreachw języku Go jest konstrukcja? Czy mogę iterować po plasterku lub tablicy za pomocą for?

tatsuhirosatou
źródło
1
Wykorzystanie rangew forpętli jest również wymienione w „interludium o typach sekcji” (w kierunku jego końca) samouczka Go.
kostix

Odpowiedzi:

851

https://golang.org/ref/spec#For_range

Instrukcja „for” z klauzulą ​​„range” iteruje wszystkie wpisy tablicy, wycinka, łańcucha lub mapy lub wartości odebranych na kanale. Dla każdego wpisu przypisuje wartości iteracji do odpowiednich zmiennych iteracji, a następnie wykonuje blok.

Jako przykład:

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

Jeśli nie zależy ci na indeksie, możesz użyć _:

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

Podkreślenie, _to pusty identyfikator , anonimowy symbol zastępczy.

davetron5000
źródło
7
W tym przykładzie elementjest wartością elementu (kopii) - to nie jest sam element. Chociaż możesz to przypisać element, nie wpłynie to na leżącą u podstaw sekwencję.
nobar
Wiem, że w Pythonie i C często używa się podkreślenia jako funkcji do lokalizacji (tj. Gettext ). Czy użycie podkreślenia spowodowałoby problemy w Go? Czy Go używa nawet tej samej biblioteki do lokalizacji?
Sergiy Kolodyazhnyy
2
@SergiyKolodyazhnyy Py docs mówi, że „(gettext) funkcja jest zwykle aliasowana jak _()w lokalnej przestrzeni nazw”, co jest zwyczajowo , nie jest częścią biblioteki lokalizacyjnej. Podkreślenie _jest prawidłową etykietą, a także konwencją w Go (i Pythonie, Scali i innych językach) do przypisywania _wartości zwracanych, których nie użyjesz. Zakres _w tym przykładzie jest ograniczony do treści forpętli. Jeśli masz funkcję _o zasięgu pakietu , byłaby ona zasłonięta wewnątrz zakresu pętli for. Jest kilka pakietów do lokalizacji, nie widziałem żadnego zastosowania _jako nazwy funkcji.
Davos,
149

Go ma foreachpodobną składnię. Obsługuje tablice / plastry, mapy i kanały.

Iteruj po tablicy lub plasterku :

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

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

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

Iteruj po mapie :

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

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

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

Iteruj po kanale :

for v := range theChan {}

Iteracja po kanale jest równoważna odbieraniu z kanału, dopóki nie zostanie zamknięty:

for {
    v, ok := <-theChan
    if !ok {
        break
    }
}
Moshe Revah
źródło
10
Chociaż OP poprosił tylko o użycie wycinka, wolę tę odpowiedź, ponieważ większość w końcu będzie potrzebować również innych zastosowań.
domoarigato,
3
ważne rozróżnienie dotyczące chanużycia: przesuwanie się po kanale z wdziękiem opuści pętlę, jeśli pisarz zamknie kanał w pewnym momencie. W for {v := <-theChan} ekwiwalent , to będzie nie wyjść na kanale końca. Możesz to sprawdzić za pomocą drugiej okwartości zwracanej. PRZYKŁAD WYCIECZKI
colm.anseo 10.10.17
Myśl ta sama podczas czytania for { ... }oznacza nieskończoną pętlę.
Lewita
13

Poniższy przykład pokazuje, jak używać rangeoperatora w forpętli do implementacji foreachpętli.

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;
}

Przykład iteruje tablicę funkcji w celu ujednolicenia obsługi błędów dla funkcji. Kompletnym przykładem jest plac zabaw Google .

PS: pokazuje również, że wiszące nawiasy klamrowe to zły pomysł na czytelność kodu. Wskazówka: forstan kończy się tuż przed action()połączeniem. Oczywiste, prawda?

Ceving
źródło
3
Dodaj ,a, gdzie forwarunek się kończy, jest jaśniejszy : play.golang.org/p/pcRg6WdxBd - To właściwie pierwszy raz, gdy znalazłem kontrargument w go fmtstylu, dzięki!
topskip
@topskip oba są poprawne; wybierz najlepszy :)
Filip Haglund,
@FilipHaglund Nie chodzi o to, czy jest ważny. Chodzi o to, że IMO jest bardziej jasne, gdzie warunek kończy się w tym konkretnym przypadku powyżej.
topskip
8
Moim zdaniem odpowiedź na to pytanie jest nieuzasadniona złożona.
AndreasHassing
@AndreasHassing Jak to zrobić zamiast wprowadzać redundancję?
ceving
10

W rzeczywistości możesz używać rangebez odwoływania się do zwracanych wartości, używając for rangeprzeciwko typowi:

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

robstarbuck
źródło
3
Dobrze wiedzieć, ale w większości przypadków nie będzie to przydatne
Sridhar
Zgodził się @Sridhar, to całkiem niszowa.
robstarbuck
9

Poniżej znajduje się przykładowy kod użycia foreach w golang

package main

import (
    "fmt"
)

func main() {

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

    for index,element := range arrayOne{

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

    }   

}

To działający przykład https://play.golang.org/p/LXptmH4X_0

Nisal Edu
źródło
bardzo dobre wytłumaczenie!
Darlan Dieterich
4

Tak, zakres :

Forma zasięgu pętli for iteruje się nad plasterkiem lub mapą.

W przypadku przejścia do wycinka dla każdej iteracji zwracane są dwie wartości. Pierwszy to indeks, a drugi to kopia elementu o tym indeksie.

Przykład:

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)
    }
}
  • Możesz pominąć indeks lub wartość, przypisując do _.
  • Jeśli chcesz tylko indeksu, całkowicie usuń wartość.
Amitesh
źródło
1

Może to być oczywiste, ale możesz wstawić tablicę w następujący sposób:

package main

import (
    "fmt"
)

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

wyjścia:

abc

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

jpihl
źródło