Klärung der Verwendung von „Defer“: Drucken des endgültigen Variablenwerts
In Go können wir mit dem Schlüsselwort „defer“ eine Funktion planen, die kurz vor dem ausgeführt wird umgebende Funktion kehrt zurück. Wenn die verzögerte Funktion jedoch lokale Variablen verwendet, die sich innerhalb der umgebenden Funktion ändern können, muss besonders darauf geachtet werden, dass die richtigen Werte verwendet werden.
Betrachten Sie die folgende Funktion:
func printNumbers() { var x int defer fmt.Println(x) for i := 0; i < 5; i++ { x++ } }
Gemäß der Sprachspezifikation werden bei der Ausführung der Defer-Anweisung die lokalen Variablen erneut ausgewertet und gespeichert. Die eigentliche Funktion wird jedoch erst später aufgerufen. In diesem Fall ist der Wert von x bei Ausführung der Defer-Anweisung 0, sodass 0 gedruckt wird, wenn die Deferred-Funktion aufgerufen wird.
Um dieses Problem zu lösen, können wir einen von mehreren Ansätzen verwenden:
1. Mit anonymer Funktion:
defer func() { fmt.Println(x) }()
Dies verwendet eine anonyme Funktion, die den aktuellen Wert von x erfasst. Wenn die anonyme Funktion aufgerufen wird, verwendet sie den erfassten Wert, auch wenn sich x in der umgebenden Funktion geändert hat.
2. Mit Zeiger:
var x int defer Print(&x)
Wir können einen Zeiger auf x verwenden und eine Funktion zurückstellen, die den gezeigten Wert ausgibt. Dies funktioniert, weil der Zeigerwert ausgewertet wird, wenn die Defer-Anweisung ausgeführt wird, der Zeigerwert jedoch erst ausgewertet wird, wenn die Deferred-Funktion aufgerufen wird.
3. Mit benutzerdefiniertem Typ:
type MyInt int func (m *MyInt) String() string { return strconv.Itoa(int(*m)) } var x MyInt defer fmt.Println(&x)
Hier erstellen wir einen benutzerdefinierten Typ MyInt, der die fmt.Stringer-Schnittstelle implementiert und es uns ermöglicht, anzugeben, wie der Wert gedruckt wird. Wenn &x gedruckt wird, wird die String-Methode aufgerufen und gibt die String-Darstellung des gezeigten Werts zurück.
4. Umschließen:
x := []int{0} defer fmt.Println(x)
Durch das Umschließen der Variablen in ein Slice können wir eine Funktion verschieben, die den Slice-Inhalt druckt. Der Slice-Deskriptor wird ausgewertet, wenn die Defer-Anweisung ausgeführt wird, aber das zugrunde liegende Array wird erst ausgewertet, wenn die Deferred-Funktion aufgerufen wird.
Es ist wichtig zu bedenken, dass die in der Deferred-Funktion verwendete Variable kein Parameter von sein sollte die verzögerte Funktion selbst, da die Parameter ausgewertet werden, wenn die Verzögerungsanweisung ausgeführt wird.
Das obige ist der detaillierte Inhalt vonWie verwende ich Defer in Go beim Umgang mit sich ändernden lokalen Variablen richtig?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!