Heim > Backend-Entwicklung > Golang > Warum gibt mein Go-Code in einem Goroutine-Datenrennen dreimal „Three' statt „One', „Two' und „Three' aus?

Warum gibt mein Go-Code in einem Goroutine-Datenrennen dreimal „Three' statt „One', „Two' und „Three' aus?

Linda Hamilton
Freigeben: 2024-12-07 22:23:13
Original
906 Leute haben es durchsucht

Why Does My Go Code Print

Das seltsame Verhalten von Goroutinen in einem Data-Race-Szenario verstehen

Im bereitgestellten Go-Code erstellen wir einen Ausschnitt aus Strukturfeldern namens Daten mit den Namen „eins“, „zwei“ und „drei“. Der Code durchläuft das Slice und erstellt Goroutinen, um den Namen jedes Felds mithilfe der Druckmethode zu drucken. Entgegen den Erwartungen gibt der Code jedoch wiederholt dreimal „drei“ aus und nicht die beabsichtigte Reihenfolge von „eins“, „zwei“ und „drei“.

Enthüllung des Datenwettlaufs

Dieses merkwürdige Verhalten ist auf einen Datenwettlauf zurückzuführen, der auftritt, wenn mehrere Goroutinen gleichzeitig auf dieselben gemeinsam genutzten Daten zugreifen und diese möglicherweise ändern. In diesem Fall entsteht das Problem durch die implizite Verwendung der Adresse der Bereichsvariablen v beim Erstellen der Goroutinen. Wenn die Schleifenvariable v in jeder Iteration geändert wird, verwenden die Goroutinen am Ende ihren Endwert, was zur konstanten Ausgabe von „drei“ führt.

Bewältigung des Datenwettlaufs

Um dieses Problem zu lösen, können wir verschiedene Ansätze verfolgen:

  • Erstellen einer neuen Variable in jeder Schleife Iteration: Innerhalb der Schleife können wir eine neue Variable mit demselben Namen wie die Bereichsvariable deklarieren und so effektiv einen neuen Gültigkeitsbereich für die Variable erstellen.

    for _, v := range data {
      v := v // Declare a new variable `v` within the loop scope.
      go v.print()
    }
    Nach dem Login kopieren
  • Ein Stück Zeiger verwenden: Anstatt ein Stück Strukturfelder zu verwenden, können wir ein Stück Zeiger auf die Felder verwenden. Dadurch wird sichergestellt, dass die Goroutinen Zeiger auf einzelne Feldelemente erhalten, wodurch Datenwettlaufprobleme vermieden werden.

    data := []*field{ {"one"},{"two"},{"three"} }
    for _, v := range data {
      go v.print()
    }
    Nach dem Login kopieren
  • Übergabe der Adresse des Slice-Elements: Eine weitere Alternative ist Übergeben Sie die Adresse jedes Elements im Slice an goroutine.

    for i := range data {
      v := &data[i] // Take the address of the slice element.
      go v.print()
    }
    Nach dem Login kopieren
  • Anonyme Funktionen verwenden und Bereichsvariablen als Argumente übergeben: Wenn sich die Goroutine-Funktion innerhalb einer anonymen Funktion befindet, können wir das Problem durch Übergabe vermeiden die Bereichsvariablen als Argumente für die Funktion.

    for _, v := range data {
      go func(v field) { // Pass the range variable `v` as an argument.
          v.print()
      }(v)
    }
    Nach dem Login kopieren

Diese Ansätze stellen sicher, dass die Goroutinen verfügen über eigene Kopien der benötigten Daten, wodurch der Datenwettlauf eliminiert wird und die korrekte Ausgabe von „eins“, „zwei“ und „drei“ in beliebiger Reihenfolge erzeugt wird.

Das obige ist der detaillierte Inhalt vonWarum gibt mein Go-Code in einem Goroutine-Datenrennen dreimal „Three' statt „One', „Two' und „Three' aus?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Quelle:php.cn
Erklärung dieser Website
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
Neueste Artikel des Autors
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage