Maison  >  Article  >  développement back-end  >  Comment implémenter des verrous à l'aide de Golang

Comment implémenter des verrous à l'aide de Golang

PHPz
PHPzoriginal
2023-04-25 16:28:392677parcourir

Golang est un langage de programmation très populaire qui prend en charge la programmation simultanée. Afin de répondre aux exigences de sécurité simultanée, Golang fournit un mécanisme de verrouillage. Un verrou est un mécanisme de synchronisation qui peut être utilisé pour contrôler l'accès aux ressources partagées. Dans cet article, nous présenterons comment implémenter des verrous à l'aide de Golang.

1. Types de verrous

Dans Golang, il existe trois principaux types de verrous : les verrous mutex, les verrous en lecture-écriture et les variables de condition.

1. Verrouillage Mutex (Mutex)

Le verrouillage Mutex est le type de verrouillage le plus simple et le plus couramment utilisé. Sa fonction est de permettre à un seul thread d'accéder aux ressources partagées en même temps. Si d'autres threads tentent d'accéder à la ressource, ils sont bloqués jusqu'à ce que le verrou soit libéré.

2. Verrouillage en lecture-écriture (RWMutex)

Le verrouillage en lecture-écriture est un autre type de verrou qui permet à plusieurs threads de lire une ressource partagée en même temps, mais n'autorise qu'un seul thread à écrire la ressource. Ce verrou est plus efficace qu'un mutex car il autorise des lectures simultanées, mais les écritures doivent occuper exclusivement la ressource.

3. Variable de condition (Cond)

La variable de condition est un mécanisme de synchronisation avancé qui fournit un mécanisme d'attente et de communication entre plusieurs threads. Les variables de condition ont deux méthodes principales : Wait et Signal. La méthode Wait peut mettre le thread en veille pour attendre des conditions spécifiques, tandis que la méthode Signal informe le thread en attente que la condition est remplie.

2. Implémentation de verrous mutex

La bibliothèque standard de Golang fournit l'implémentation de verrous mutex. Vous pouvez utiliser le type Mutex dans le package de synchronisation pour implémenter des verrous mutex. Voici un exemple de programme :

package main

import (
    "fmt"
    "sync"
)

var (
    count int
    lock  sync.Mutex
)

func main() {
    var wg sync.WaitGroup
    for i := 0; i < 100; i++ {
        wg.Add(1)
        go func() {
            lock.Lock()
            count++
            lock.Unlock()
            wg.Done()
        }()
    }
    wg.Wait()
    fmt.Println(count)
}

Dans le code ci-dessus, nous définissons un compteur et créons un verrou mutex. Ensuite, 100 goroutines sont démarrées et le compteur est incrémenté de 1 dans chaque goroutine. Étant donné que l'accès au décompte est simultané, nous devons utiliser un verrou mutex pour éviter que des conditions de concurrence critique ne se produisent.

Il convient de noter que pour les opérations impliquant des ressources partagées, l'opération doit être effectuée après l'acquisition du verrou, et le verrou doit être libéré une fois l'opération terminée pour garantir que toutes les opérations sont atomiques.

3. Implémentation du verrouillage en lecture-écriture

Semblable aux verrous mutex, la bibliothèque standard de Golang fournit également l'implémentation de verrous en lecture-écriture. Vous pouvez utiliser le type RWMutex dans le package de synchronisation pour implémenter des verrous en lecture-écriture. Voici un exemple de programme :

package main

import (
    "fmt"
    "sync"
)

var (
    count int
    rw    sync.RWMutex
)

func main() {
    var wg sync.WaitGroup
    for i := 0; i < 100; i++ {
        wg.Add(1)
        go func() {
            rw.Lock()
            count++
            rw.Unlock()
            wg.Done()
        }()
    }
    for i := 0; i < 100; i++ {
        wg.Add(1)
        go func() {
            rw.RLock()
            fmt.Println(count)
            rw.RUnlock()
            wg.Done()
        }()
    }
    wg.Wait()
}

Dans le code ci-dessus, nous utilisons le type RWMutex pour implémenter des verrous en lecture-écriture. Tout d'abord, démarrez 100 goroutines pour ajouter 1 au compteur. Pendant cette période, le compteur ne peut être occupé que par un seul thread. Ensuite, démarrez 100 goroutines pour lire la valeur du compteur. Ces goroutines peuvent lire la valeur du compteur en même temps.

Par rapport aux verrous mutex, les verrous en lecture-écriture ont une concurrence plus élevée et moins de conflits de verrouillage. Si la plupart des opérations sont des opérations de lecture, l'utilisation de verrous en lecture-écriture au lieu de verrous mutex peut améliorer les performances.

4. Implémentation de variables de condition

Les variables de condition sont un mécanisme de synchronisation avancé qui peut mettre les threads en veille et attendre que des conditions spécifiques soient remplies. Les variables de condition sont également fournies dans le package de synchronisation et peuvent être implémentées à l'aide du type Cond. Voici un exemple de programme :

package main

import (
    "fmt"
    "sync"
)

var (
    count   int
    waiters int
    lock    sync.Mutex
    cond    *sync.Cond = sync.NewCond(&lock)
)

func main() {
    var wg sync.WaitGroup
    for i := 0; i < 10; i++ {
        wg.Add(1)
        go func() {
            lock.Lock()
            for count < 5 {
                waiters++
                cond.Wait()
                waiters--
            }
            fmt.Println("Count:", count)
            lock.Unlock()
            wg.Done()
        }()
    }
    for i := 0; i < 5; i++ {
        wg.Add(1)
        go func() {
            lock.Lock()
            count++
            if waiters > 0 {
                cond.Signal()
            }
            lock.Unlock()
            wg.Done()
        }()
    }
    wg.Wait()
}

Dans le code ci-dessus, nous définissons un compteur et un compteur de threads en attente. Ensuite, démarrez 10 goroutines pour attendre que le compteur atteigne 5 et affichez la valeur du compteur lorsque le compteur atteint 5. Démarrez 5 goroutines supplémentaires pour en ajouter 1 au compteur. Lorsque le compteur atteint 5, appelez cond.Signal() pour réveiller la goroutine en attente.

Il convient de noter que le verrou mutex doit être acquis avant d'utiliser la variable de condition, sinon un blocage se produira.

5. Résumé

Cet article présente trois types de verrous courants dans Golang : les verrous mutex, les verrous en lecture-écriture et les variables de condition. Les verrous mutex sont les verrous les plus couramment utilisés pour empêcher plusieurs threads d'accéder simultanément aux ressources partagées. Les verrous en lecture-écriture autorisent plusieurs opérations de lecture, mais uniquement des opérations d'écriture exclusives. Les variables de condition sont un mécanisme de synchronisation avancé qui permet aux threads d'attendre que des conditions spécifiques soient remplies.

Lors de l'écriture de programmes Golang, le choix du type de verrouillage approprié est crucial pour garantir la sécurité de la concurrence. Il est nécessaire de sélectionner le type de verrou approprié en fonction du scénario d'application et des besoins spécifiques pour améliorer les performances de concurrence.

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn