„Wenn ein Arbeiter seine Arbeit gut machen will, muss er zuerst seine Werkzeuge schärfen.“ – Konfuzius, „Die Gespräche des Konfuzius. Lu Linggong“
Titelseite > Programmierung > Benchmark mit Golang

Benchmark mit Golang

Veröffentlicht am 04.11.2024
Durchsuche:132

Hallo Gophers ?

In diesem Blogbeitrag zeige ich Ihnen, wie Sie ein fantastisches Tool verwenden, das im #golang-Testpaket integriert ist. Wie würden Sie die Leistung eines Codeteils oder einer Funktion testen? Verwenden Sie Benchmark-Tests.

Lass uns gehen.

Für diesen Test verwende ich die klassische Fibonacci-Zahl oder Fibonacci-Folge, die bestimmt wird durch:

if (x 



Diese Reihenfolge ist wichtig, da sie auch in mehreren Teilen der Mathematik und der Natur vorkommt, wie unten gezeigt:

Benchmark with Golang

Es gibt mehrere Möglichkeiten, diesen Code zu implementieren, und ich werde zwei für unsere Benchmark-Tests auswählen: die rekursiven und iterativen Methoden zur Berechnung. Das Hauptziel der Funktionen besteht darin, eine Position bereitzustellen und die Fibonacci-Zahl an dieser Position zurückzugeben.

Rekursive Methode

// main.go

func fibRecursive(n int) int {
    if n 



Iterative Methode

// main.go

func fibIterative(position uint) uint {
    slc := make([]uint, position)
    slc[0] = 1
    slc[1] = 1

    if position 



Diese Methoden sind nicht optimiert, aber die Ergebnisse der Tests unterscheiden sich selbst bei einer kleinen Anzahl erheblich. Das werden Sie in den Tests sehen. Um dem Code zu folgen, klicken Sie hier.

Lassen Sie uns nun für die Benchmark-Tests einige Tests in die Datei _main_test.go schreiben. Mithilfe der Golang-Dokumentation zum Benchmark können Sie die zu testenden Funktionen wie folgt erstellen:

// main_test.go

// The key is to start every function you want to benchmark with the keyword Benchmark and use b *testing.B instead of t *testing.T as input 
func BenchmarkFibIterative(b *testing.B) {
    // Use this for-loop to ensure the code will behave correctly. 
    // Now, you can put the function or piece of code you want to test
    for i := 0; i 



Frage, bevor Sie fortfahren: Welches ist schneller?

Lassen Sie uns den Test für eine kleine Zahl (10) und für eine etwas größere Zahl (80) durchführen. Um die Benchmark-Tests auszuführen, führen Sie einfach den Befehl aus:

go test -bench=NameoftheFunction

Wenn Sie mehr über diesen Befehl erfahren möchten, schauen Sie hier nach.

Erster Test: Position=10

//(fibIterative)
Results:
cpu: Intel(R) Core(TM) i7-7700HQ CPU @ 2.80GHz
BenchmarkFibIterative-8         24491042                42.50 ns/op
PASS
ok      playground      1.651s

Lassen Sie uns anhand dieses Bildes analysieren:

Benchmark with Golang

Dem Bild zufolge haben wir 8 Kerne für die Tests, keine zeitliche Begrenzung (es läuft bis zum Abschluss). Es dauerte 1,651 Sekunden, um die Aufgabe abzuschließen.

==== Extra ====
We got 24,491,042 iterations (computations), and each iteration (op) took 42.50 ns.

Doing some math, we can calculate how much time one op took:

42.50 ns/op with 1 ns = 1/1,000,000,000 s
op ≈ 2.35270590588e-12 s
==== Extra ====

Das ist ein gutes Ergebnis. Schauen wir uns die rekursive Funktion für Position 10 an:

// Results
BenchmarkFibRecursive-8          6035011               187.8 ns/op
PASS
ok      playground      1.882s

Wir können sehen, dass es 1,882 Sekunden dauerte, um die Aufgabe abzuschließen.

Die iterative Funktion hat mit wenigen Dezisekunden Vorsprung gewonnen. Versuchen wir noch einen Test mit:

Position 50

// Results for the Iterative Function

cpu: Intel(R) Core(TM) i7-7700HQ CPU @ 2.80GHz
BenchmarkFibIterative-8         27896118                45.37 ns/op
PASS
ok      playground      2.876s

// Results for the Recursive Function

cpu: Intel(R) Core(TM) i7-7700HQ CPU @ 2.80GHz
BenchmarkFibRecursive-8          6365198               186.3 ns/op
PASS
ok      playground      1.918s

Wow! Jetzt ist die rekursive Funktion schneller?

Lassen Sie uns mit einer etwas größeren Zahl abschließen.

Position 80

// Results for the Iterative Function

cpu: Intel(R) Core(TM) i7-7700HQ CPU @ 2.80GHz
BenchmarkFibIterative-8          5102344               229.5 ns/op
PASS
ok      playground      1.933s

// Results for the Recursive Function
// My poor PC couldn’t handle it, so I had to reduce the position to 50 just to get some results printed.

cpu: Intel(R) Core(TM) i7-7700HQ CPU @ 2.80GHz
BenchmarkFibRecursive-8                1        44319299474 ns/op
PASS
ok      playground      44.951s

Der Unterschied ist riesig. Für Position 80 dauerte der iterative Ansatz etwa 2 Sekunden. Für Position 50 dauerte die rekursive Funktion etwa 45 Sekunden. Dies zeigt, wie wichtig es ist, Ihren Code zu vergleichen, wenn Ihr Golang-Projekt langsamer wird.

Abschluss

Wenn Ihr Produktionscode langsam läuft oder unvorhersehbar langsamer ist, können Sie diese Technik in Kombination mit pprof oder anderen Tools aus dem integrierten Testpaket verwenden, um die Leistung Ihres Codes zu identifizieren und zu testen schlecht und wie man es optimiert.

Randbemerkung: Nicht jeder Code, der schön für das Auge ist, ist leistungsfähiger.

Zusätzliche Übung

Können Sie einen besseren Weg finden, die rekursive Funktion zu verbessern? (Tipp: Verwenden Sie dynamische Programmierung). In diesem Artikel wird erklärt, warum für einige kleine Zahlen die rekursive Strategie besser ist.

Freigabeerklärung Dieser Artikel ist abgedruckt unter: https://dev.to/pedrobertao/benchmark-with-golang-md4?1 Bei Verstößen wenden Sie sich bitte an [email protected], um ihn zu löschen
Neuestes Tutorial Mehr>

Haftungsausschluss: Alle bereitgestellten Ressourcen stammen teilweise aus dem Internet. Wenn eine Verletzung Ihres Urheberrechts oder anderer Rechte und Interessen vorliegt, erläutern Sie bitte die detaillierten Gründe und legen Sie einen Nachweis des Urheberrechts oder Ihrer Rechte und Interessen vor und senden Sie ihn dann an die E-Mail-Adresse: [email protected] Wir werden die Angelegenheit so schnell wie möglich für Sie erledigen.

Copyright© 2022 湘ICP备2022001581号-3