Heim  >  Artikel  >  Backend-Entwicklung  >  Einführung in Go Defer (Go-Delay-Funktion)

Einführung in Go Defer (Go-Delay-Funktion)

尚
nach vorne
2019-11-30 14:03:382284Durchsuche

Die Verzögerung von

Einführung in Go Defer (Go-Delay-Funktion)

Go-Sprache gilt als neue Funktion der Sprache, zumindest im Vergleich zu den heutigen Mainstream-Programmiersprachen.

Die Defer-Anweisung ruft eine Funktion auf, bis die periphere Funktion zurückkehrt oder die periphere Funktion bis zum Ende ausgeführt wird oder die entsprechende Goroutine in Panik gerät

Wann immer defer ist ausgeführt, es Der nachfolgende Funktionswert (in Go ist die Funktion ein Referenztyp, ein Bürger erster Klasse und kann einer Variablen zugewiesen werden) und Funktionsparameter werden ausgewertet, aber die Funktion wird erst sofort aufgerufen ( ↑ ) treten die oben genannten drei Situationen auf. Dies ist der gesamte Inhalt von defer, weg, der Rest ist die beste Praxis von defer

Die Funktion wird nicht sofort aufgerufen

Beginnen wir mit der einfachsten:

func readFile(fileName string){
    f,err := os.Open(fileName)
    if err!=nil {
        return
    }
    defer f.Close()
    var content [1024]byte
    f.Read(content[:])
    fmt.Printf("%s",content)
}
func main() {
    readFile("test.data")
}

Das Programm gibt die ersten 1024 Bytes von test.data aus. Es ist erwähnenswert, dass Open/Close-Pairing-Operationen wie diese die idiomatische Verwendung von defer darstellen. Dieses Beispiel erklärt die zweite Hälfte des obigen Satzes

„Aber die Funktion wird nicht aufgerufen“

Denn wenn f.Close() nach der Verzögerung nicht verzögert wird, dann ist der Dateideskriptor Wenn sie sind alle geschlossen, es wird nichts gelesen.

Funktionswert und Funktionsparameter werden ausgewertet, aber die Funktion wird nicht sofort aufgerufen

Das folgende Beispiel erklärt die erste Hälfte, sie stammt aus a8093152e673feb7aba1828c43532094 , leichte Modifikation:

func trace(funcName string) func(){
    start := time.Now()
    fmt.Printf("function %s enter\n",funcName)
    return func(){
        log.Printf("function %s exit (elapsed %s)",funcName,time.Since(start))
    }
}
 
func foo(){
    defer trace("foo()")()
    time.Sleep(5*time.Second)
}
func main(){
    foo()
    foo()
}
/*
OUTPUT:
function foo() enter
function foo() exit (elapsed 5.0095471s)
function foo() enter
function foo() exit (elapsed 5.0005382s)
*/

Warum wird die Ausgabe von foo eingegeben und dann etwa fünf Sekunden gewartet, bevor die Ausgabe beendet wird? Denn wie gesagt, die Funktionswerte und Parameter nach

defer werden ausgewertet aber die eigentliche Funktion Der Aufruf muss bis zum Ende warten

Der Funktionswert ist hier die anonyme Funktion, die von Trace() zurückgegeben wird. Der Funktionsparameter ist natürlich der String-Literalwert „foo()“. Von Trace("foo()") wird die Ausgabefunktion foo() eingegeben, der eigentliche Funktionsaufruf von Trace("foo()")(), das heißt, die Ausgabefunktion foo() exit(elapsed x.x) wird verschoben bis Return-Ausführung (wenn Return die Rückgabewertvariable aktualisiert, wird sie nach der Aktualisierung ausgeführt, bevor die Defer-Funktion ausgeführt wird).

Verschiedenes

Ein bisschen mehr: Wenn mehrere Defer-Anweisungen vorhanden sind, ist die Ausführungsreihenfolge der letzten Defer-Funktion entgegengesetzt zu der Reihenfolge, in der Defer angezeigt wird, z :

func main() {
    func1 := func(){
        fmt.Println("func1() execution deferred")
    }
    func2 := func(){
        fmt.Println("func2() execution deferred")
    }
    defer func1()
    defer func2()
    fmt.Println("strat\nworking...")
}
/*
OUTPUT:
strat
working...
func2() execution deferred
func1() execution deferred
*/

Das obige ist der detaillierte Inhalt vonEinführung in Go Defer (Go-Delay-Funktion). Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Dieser Artikel ist reproduziert unter:cnblogs.com. Bei Verstößen wenden Sie sich bitte an admin@php.cn löschen