Heim >Backend-Entwicklung >Golang >Go Concurrency: Mutexe vs. Kanäle mit Beispielen

Go Concurrency: Mutexe vs. Kanäle mit Beispielen

Mary-Kate Olsen
Mary-Kate OlsenOriginal
2025-01-08 20:10:54543Durchsuche

Zählersynchronisation in der gleichzeitigen Go-Programmierung: Mutex, gepufferte Kanäle und ungepufferte Kanäle

Beim Erstellen gleichzeitiger Anwendungen in Go ist die Synchronisierung von entscheidender Bedeutung, um einen sicheren Zugriff auf gemeinsam genutzte Daten zu gewährleisten. Mutex und Channel sind die Hauptwerkzeuge für die Synchronisierung in Go.

In diesem Artikel werden verschiedene Möglichkeiten zum Erstellen sicherer Parallelitätszähler untersucht. Während der Referenzartikel dieses Problem mit Mutex löst, werden wir auch Alternativen mit gepufferten und ungepufferten Kanälen untersuchen.

Problembeschreibung

Wir müssen einen Zähler bauen, der gleichzeitig sicher verwendet werden kann.

Zählercode

<code class="language-go">package main

type Counter struct {
    count int
}

func (c *Counter) Inc() {
    c.count++
}

func (c *Counter) Value() int {
    return c.count
}</code>

Um unsere Code-Parallelität sicher zu machen, schreiben wir einige Tests.

1. Verwenden Sie Mutex

Mutex (Mutex) ist ein Synchronisationsprimitiv, das sicherstellt, dass jeweils nur eine Goroutine auf kritische Teile des Codes zugreifen kann. Es bietet einen Sperrmechanismus: Wenn eine Goroutine Mutex sperrt, werden andere Goroutinen, die versuchen, sie zu sperren, blockiert, bis Mutex entsperrt wird. Daher wird es häufig verwendet, wenn eine gemeinsam genutzte Variable oder Ressource vor Race Conditions geschützt werden muss.

<code class="language-go">package main

import (
    "sync"
    "testing"
)

func TestCounter(t *testing.T) {
    t.Run("using mutexes and wait groups", func(t *testing.T) {
        counter := Counter{}
        wantedCount := 1000

        var wg sync.WaitGroup
        var mut sync.Mutex

        wg.Add(wantedCount)

        for i := 0; i < wantedCount; i++ {
            go func() {
                defer wg.Done()
                mut.Lock()
                counter.Inc()
                mut.Unlock()
            }()
        }

        wg.Wait()
        if counter.Value() != wantedCount {
            t.Errorf("got %d, want %d", counter.Value(), wantedCount)
        }
    })
}</code>

Der Code verwendet sync.WaitGroup, um den Abschluss aller Goroutinen zu verfolgen, und verwendet sync.Mutex, um zu verhindern, dass mehrere Goroutinen gleichzeitig auf den gemeinsamen Zähler zugreifen.

2. Pufferkanal verwenden

Go Concurrency: Mutexes vs Channels with Examples

Kanäle sind für Go eine Möglichkeit, Goroutinen eine sichere Kommunikation zu ermöglichen. Sie sind in der Lage, Daten zwischen Goroutinen zu übertragen und für Synchronisierung zu sorgen, indem sie den Zugriff auf die übergebenen Daten kontrollieren.

In diesem Beispiel verwenden wir Kanäle, um Goroutinen zu blockieren und nur einer Goroutine den Zugriff auf die gemeinsam genutzten Daten zu ermöglichen. Gepufferte Kanäle haben eine feste Kapazität, das heißt, sie können eine vordefinierte Anzahl von Elementen aufnehmen, bevor sie den Absender blockieren. Der Absender blockiert nur, wenn der Puffer voll ist.

<code class="language-go">package main

import (
    "sync"
    "testing"
)

func TestCounter(t *testing.T) {
    t.Run("using buffered channels and wait groups", func(t *testing.T) {
        counter := Counter{}
        wantedCount := 1000

        var wg sync.WaitGroup
        wg.Add(wantedCount)

        ch := make(chan struct{}, 1)

        ch <- struct{}{} // 允许第一个 goroutine 开始

        for i := 0; i < wantedCount; i++ {
            go func() {
                defer wg.Done()
                <-ch
                counter.Inc()
                ch <- struct{}{}
            }()
        }

        wg.Wait()
        if counter.Value() != wantedCount {
            t.Errorf("got %d, want %d", counter.Value(), wantedCount)
        }
    })
}</code>

Der Code verwendet einen gepufferten Kanal mit einer Kapazität von 1, sodass jeweils nur eine Goroutine auf den Zähler zugreifen kann.

3. Verwenden Sie nicht gepufferte Kanäle

Go Concurrency: Mutexes vs Channels with Examples

Ungepufferte Kanäle haben keine Puffer. Sie blockieren den Sender, bis der Empfänger bereit ist, Daten zu empfangen. Dies sorgt für eine strikte Synchronisierung, bei der Daten einzeln zwischen Goroutinen übertragen werden.

<code class="language-go">package main

import (
    "sync"
    "testing"
)

func TestCounter(t *testing.T) {
    t.Run("using unbuffered channels and wait groups", func(t *testing.T) {
        counter := Counter{}
        wantedCount := 1000

        var wg sync.WaitGroup
        wg.Add(wantedCount)

        ch := make(chan struct{})

        go func() {
            for i := 0; i < wantedCount; i++ {
                ch <- struct{}{}
            }
            close(ch)
        }()

        for range ch {
            counter.Inc()
            wg.Done()
        }

        if counter.Value() != wantedCount {
            t.Errorf("got %d, want %d", counter.Value(), wantedCount)
        }
    })
}
</code>

Der Code verwendet ungepufferte Kanäle, um sicherzustellen, dass jeweils nur eine Goroutine auf den Zähler zugreift.

4. Pufferkanal anstelle von WaitGroup verwenden

Wir können auch gepufferte Kanäle ohne WaitGroup verwenden, beispielsweise mithilfe einer Endlosschleife oder eines anderen Kanals, um den Abschluss der Goroutine zu verfolgen.

Fazit

In diesem Artikel werden verschiedene Ansätze zum Erstellen sicherer Parallelitätszähler in Go untersucht. Die Kenntnis dieser Tools und deren Verwendung ist der Schlüssel zum Schreiben effizienter und sicherer gleichzeitiger Go-Programme.

Referenzressourcen

Dieser Artikel ist vom Synchronisationskapitel in „Learn Go with Tests“ inspiriert.

Ich hoffe, dieser Artikel ist hilfreich für Sie!

Das obige ist der detaillierte Inhalt vonGo Concurrency: Mutexe vs. Kanäle mit Beispielen. 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