Czy foreach
w języku Go jest konstrukcja? Czy mogę iterować po plasterku lub tablicy za pomocą for
?
range
w for
pętli jest również wymienione w „interludium o typach sekcji” (w kierunku jego końca) samouczka Go.
Czy foreach
w języku Go jest konstrukcja? Czy mogę iterować po plasterku lub tablicy za pomocą for
?
range
w for
pętli jest również wymienione w „interludium o typach sekcji” (w kierunku jego końca) samouczka Go.
Odpowiedzi:
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.
element
jest 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ę.
_()
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 for
pę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.
Go ma foreach
podobną 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
}
}
chan
uż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 ok
wartości zwracanej. PRZYKŁAD WYCIECZKI
for { ... }
oznacza nieskończoną pętlę.
Poniższy przykład pokazuje, jak używać range
operatora w for
pętli do implementacji foreach
pę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: for
stan kończy się tuż przed action()
połączeniem. Oczywiste, prawda?
,
a, gdzie for
warunek się kończy, jest jaśniejszy : play.golang.org/p/pcRg6WdxBd - To właściwie pierwszy raz, gdy znalazłem kontrargument w go fmt
stylu, dzięki!
W rzeczywistości możesz używać range
bez odwoływania się do zwracanych wartości, używając for range
przeciwko 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++
}
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
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ż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