Heim > Backend-Entwicklung > Golang > Was führt dazu, dass meine Goroutine im folgenden Mutex-Code hängen bleibt?

Was führt dazu, dass meine Goroutine im folgenden Mutex-Code hängen bleibt?

王林
Freigeben: 2024-02-13 18:57:07
nach vorne
714 Leute haben es durchsucht

是什么导致我的 goroutine 在以下互斥体代码中陷入僵局?

PHP-Editor Wettbewerb um gemeinsame Ressourcen. Wenn jedoch Probleme im Code auftreten, kann dies dazu führen, dass die Goroutine blockiert und die Ausführung nicht fortgesetzt werden kann. Als nächstes werden wir die möglichen Ursachen dieses Problems im Detail besprechen und Lösungen anbieten.

Frageninhalt

Ich versuche eine Schlüsselkarte zu speichern, jeder Schlüssel hat ein separates Schloss. Beim Erstellen einer Sperre für einen bestimmten Schlüssel verwende ich einen globalen Mutex, um in die Karte zu schreiben.

Nachdem ich das Schloss für den Schlüssel fertig erstellt habe, werde ich das neue Schloss verwenden und es freigeben, wenn die Arbeit erledigt ist. Derzeit versuche ich, einen einzelnen Schlüssel zu ändern, um meinen Code zu testen.

Das ist der Code:

// You can edit this code!
// Click here and start typing.
package main

import (
    "fmt"
    "sync"
    "time"
)

var count int
var globalMutex *sync.RWMutex
var mutexes map[int]*sync.Mutex

func MyLock(index int) {
    fmt.Println("Aquiring Lock")
    globalMutex.Lock()
    defer globalMutex.Unlock()

    count++

    if mutexes == nil {
        mutexes = make(map[int]*sync.Mutex)
    }

    if _, ok := mutexes[index]; !ok {
        mutexes[index] = &sync.Mutex{}
    }

    fmt.Println("Aquiring 2nd Lock")
    mutexes[index].Lock()
    fmt.Println("Aquired Lock")
}

func MyUnlock(index int) {
    globalMutex.Lock()
    defer globalMutex.Unlock()
    mutexes[index].Unlock()
}

func main() {
    var wg sync.WaitGroup
    globalMutex = &sync.RWMutex{}
    wg.Add(500)
    for i := 0; i < 500; i++ {
        go func(i int) {
            defer wg.Done()

            MyLock(2)

            time.Sleep(1 * time.Second)
            fmt.Println(i)

            MyUnlock(2)
        }(i)
    }

    wg.Wait()
    fmt.Println(mutexes)
    fmt.Println(count)
}
Nach dem Login kopieren

Ich bin mir nicht sicher, warum die Sperre nicht aktiviert werden kann. Spielplatz-Link: https://go.dev/play/p/-co0xaxpuy0

Solution

mylock kann globale Mutexe und einzelne Mutexe sperren. Dies macht das Entsperren in manchen Fällen unmöglich:

  1. goroutine 1 ruft mylock(2) auf. Dadurch wird die einzelne Sperre l2 wie erwartet gesperrt.
  2. goroutine 2 ruft mylock(2) auf. Es erhält die globale Sperre und blockiert dann, bis l2 freigegeben wird. Während des Wartens bleibt die globale Sperre bestehen.
  3. goroutine 1 ruft myunlock(2) auf. Es ist blockiert und wartet darauf, dass die globale Sperre (gehalten von Goroutine 2) aufgehoben wird. Das ist eine Sackgasse.

Um dies zu beheben, geben Sie die einzelnen Sperren zurück, während Sie die globale Sperre beibehalten, anstatt sie zu (ent)sperren. myunlock-Funktion wird überflüssig: ​​

func mylock(index int) *sync.mutex {
    globalmutex.lock()
    defer globalmutex.unlock()

    count++

    if mutexes == nil {
        mutexes = make(map[int]*sync.mutex)
    }

    mu := mutexes[index]
    if mu == nil {
        mu = &sync.mutex{}
        mutexes[index] = mu
    }

    return mu
}

func main() {
    var wg sync.waitgroup
    globalmutex = &sync.rwmutex{}
    wg.add(500)
    for i := 0; i < 500; i++ {
        go func(i int) {
            defer wg.done()

            mu := mylock(2)
            mu.lock()
            defer mu.unlock()

            time.sleep(1 * time.second)
            fmt.println(i)
        }(i)
    }

    wg.wait()
    fmt.println(mutexes)
    fmt.println(count)
}
Nach dem Login kopieren

Um die Leistung zu verbessern, können Sie zunächst prüfen, ob eine einzelne Sperre vorhanden ist, während Sie nur eine globale Lesesperre halten (beachten Sie, dass sich dadurch ändert, was count darstellt):

func MyLock(index int) *sync.Mutex {
    globalMutex.RLock()
    mu := mutexes[index]
    globalMutex.RUnlock()

    if mu != nil {
        return mu
    }
    
    globalMutex.Lock()
    defer globalMutex.Unlock()

    count++

    if mutexes == nil {
        mutexes = make(map[int]*sync.Mutex)
    }

    mu = mutexes[index] // have to check again because
    if mu == nil {      // we briefly released globalMutex 
        mu = &sync.Mutex{}
        mutexes[index] = mu
    }

    return mu
}
Nach dem Login kopieren

Das obige ist der detaillierte Inhalt vonWas führt dazu, dass meine Goroutine im folgenden Mutex-Code hängen bleibt?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Quelle:stackoverflow.com
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
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage