Heim >Backend-Entwicklung >Golang >Wie geht Go mit erfassten Schleifenvariablen in Abschlüssen um und warum ist das wichtig?

Wie geht Go mit erfassten Schleifenvariablen in Abschlüssen um und warum ist das wichtig?

Mary-Kate Olsen
Mary-Kate OlsenOriginal
2024-12-15 11:56:10865Durchsuche

How Does Go Handle Captured Loop Variables in Closures, and Why Does it Matter?

Erfasster Abschluss (für Schleifenvariable) in Go

Grundlegendes zum Abschluss von Schleifenvariablen

In Go kann der Compiler automatisch Schleifenvariablen zur Verwendung darin erfassen Schließungen, aber dieses Verhalten variiert je nach Art der Schleife. In for...range-Schleifen werden die Schleifenvariablen als Referenzen auf die Iterationsvariablen der äußeren Schleife erfasst.

Grund für Referenzabschlüsse in for...range-Schleifen

Go treats for.. .range-Schleifen und andere for-Schleifen auf ähnliche Weise. Daher verweist der erfasste Abschluss für eine Schleifenvariable in einer for...range-Schleife auf denselben Speicherort wie die Variable der äußeren Schleife.

Konsequenzen von Referenzabschlüssen

In diesem Szenario werden alle Änderungen vorgenommen Änderungen an der Variablen des erfassten Abschlusses wirken sich auch auf die Variable der äußeren Schleife aus und führen möglicherweise zu unerwartetem Verhalten. Um dieses Problem zu vermeiden, ist es notwendig, eine Kopie der Schleifenvariablen innerhalb des Abschlusses zu erstellen, wie im Beispiel „Wert...Bereich“ unten gezeigt.

Beispielcode

Der bereitgestellte Code Das Snippet veranschaulicht den Unterschied zwischen der Erfassung einer Referenz auf die Schleifenvariable und der Erfassung derselben Wert:

func main() {
    lab1() // captured closure is not what is expected

    lab2() // captured closure is not what is expected

    lab3() // captured closure behaves ok
}

func lab3() {
    m := make(map[int32]int32)
    for i := 1; i <= 10; i++ {
        m[i] = i
    }

    l := [](func() (int32, int32)){}
    for k, v := range m {
        kLocal, vLocal := k, v // (C) captures just the right values assigned to k and v
        l = append(l, func() (int32, int32) {
            return kLocal, vLocal
        })
    }

    for _, x := range l {
        k, v := x()
        fmt.Println(k, v)
    }
}

func lab2() {
    m := make(map[int32]int32)
    for i := 1; i <= 10; i++ {
        m[i] = i
    }

    l := [](func() (int32, int32)){}
    for k, v := range m {
        l = append(l, func() (int32, int32) {
            kLocal, vLocal := k, v // (B) captures just the last values assigned to k and v from the range
            return kLocal, vLocal
        })
    }

    for _, x := range l {
        k, v := x()
        fmt.Println(k, v)
    }
}

func lab1() {
    m := make(map[int32]int32)
    for i := 1; i <= 10; i++ {
        m[i] = i
    }

    l := [](func() (int32, int32)){}
    for k, v := range m {
        l = append(l, func() (int32, int32) { return k, v }) // (A) captures just the last values assigned to k and v from the range
    }

    for _, x := range l {
        k, v := x()
        fmt.Println(k, v)
    }
}

Ausgabe

In lab1 verweisen die erfassten Abschlüsse auf die Endwerte aus der Schleife statt auf die erwarteten Einzelwerte. In Lab2 verweisen die Captures weiterhin auf die endgültigen Werte, da der erstellte Abschluss dieselben Schleifenvariablen verwendet, auf die an anderer Stelle im äußeren Bereich verwiesen wird. In lab3 erfassen die Abschlüsse Kopien der Schleifenvariablen, sodass sie die einzelnen Werte genau darstellen.

Das obige ist der detaillierte Inhalt vonWie geht Go mit erfassten Schleifenvariablen in Abschlüssen um und warum ist das wichtig?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn