Maison  >  Article  >  développement back-end  >  Comment résoudre le problème de blocage en langage Go ?

Comment résoudre le problème de blocage en langage Go ?

PHPz
PHPzoriginal
2023-10-08 17:07:411040parcourir

Comment résoudre le problème de blocage en langage Go ?

Comment résoudre le problème de blocage en langage Go ?

Le langage Go a les caractéristiques de la programmation simultanée, et des opérations simultanées peuvent être réalisées en utilisant goroutine et canal. Cependant, le blocage est un problème courant en programmation simultanée. Lorsque les goroutines dépendent les unes des autres et créent des dépendances circulaires lors de l'accès à ces ressources, des blocages peuvent survenir. Cet article explique comment résoudre le problème de blocage dans le langage Go et fournit des exemples de code spécifiques.

Tout d’abord, comprenons ce qu’est une impasse. Le blocage fait référence à deux processus ou plus (ou goroutines) qui attendent indéfiniment des ressources occupées les uns par les autres, empêchant le programme de poursuivre son exécution. Dans le langage Go, un blocage se produit généralement lors du processus de communication entre les goroutines, entraînant une attente mutuelle en raison de conditions de concurrence ou d'une utilisation incorrecte des verrous.

Ce qui suit est un exemple simple qui démontre l'apparition du problème de blocage :

package main

import "fmt"

func main() {
    ch := make(chan int)
    ch <- 1
    fmt.Println(<-ch)
}

Dans le code ci-dessus, nous créons un canal sans tampon (ch), puis envoyons l'entier 1 au canal dans la goroutine (ch < ; - 1), puis recevez immédiatement les données du canal (

La clé pour résoudre le problème de blocage est d'éviter les dépendances circulaires et d'utiliser correctement les mécanismes de synchronisation, tels que les mutex et les variables de condition. Voici plusieurs solutions courantes :

  1. Exécution asynchrone : utilisez un canal mis en mémoire tampon ou utilisez une instruction select pour effectuer un traitement non bloquant des opérations de lecture et d'écriture du canal, ce qui peut éviter les blocages entre les opérations d'envoi et de réception.
package main

import "fmt"

func main() {
    ch := make(chan int, 1)

    go func() {
        ch <- 1
    }()

    fmt.Println(<-ch)
}

Dans le code ci-dessus, nous utilisons un canal tamponné (ch) pour éviter le blocage, afin que l'opération d'envoi (ch

  1. Utilisez les verrous mutex : les verrous mutex peuvent protéger les ressources partagées et empêcher plusieurs goroutines d'y accéder en même temps. Avant d'accéder aux ressources partagées, vous devez acquérir le verrou, puis le libérer après utilisation pour éviter un blocage.
package main

import "fmt"
import "sync"

func main() {
    var wg sync.WaitGroup
    var mu sync.Mutex
    x := 0

    for i := 0; i < 100; i++ {
        wg.Add(1)
        go func() {
            mu.Lock()
            defer mu.Unlock()
            x++
            wg.Done()
        }()
    }

    wg.Wait()
    fmt.Println(x)
}

Dans le code ci-dessus, nous utilisons un verrou mutex (mu) pour protéger la ressource partagée (x) et garantir qu'il n'y a qu'une seule goroutine via le verrouillage (mu.Lock()) et le déverrouillage (mu.Unlock( )) opérations Possibilité d'accéder à des ressources partagées.

  1. Utiliser des variables de condition : l'utilisation de variables de condition permet à goroutine d'attendre ou de se réveiller lorsque des conditions spécifiques sont remplies. En utilisant des variables de condition, une logique de synchronisation complexe peut être implémentée pour éviter efficacement les blocages.
package main

import "fmt"
import "sync"

func main() {
    var wg sync.WaitGroup
    var mu sync.Mutex
    cond := sync.NewCond(&mu)
    x := 0
    flag := false

    for i := 0; i < 100; i++ {
        wg.Add(1)
        go func() {
            mu.Lock()
            defer mu.Unlock()

            for !flag {
                cond.Wait()
            }

            x++
            wg.Done()
        }()
    }

    mu.Lock()
    flag = true
    cond.Broadcast()
    mu.Unlock()

    wg.Wait()
    fmt.Println(x)
}

Dans le code ci-dessus, nous utilisons la variable de condition (cond) pour attendre ou réveiller la goroutine, et effectuons l'opération (x++) lorsque la condition est remplie (l'indicateur est vrai). Attendez que la condition se produise en appelant cond.Wait() et utilisez cond.Broadcast() pour réveiller la goroutine en attente.

Pour résumer, résoudre le problème de blocage dans le langage Go nécessite d'éviter les dépendances circulaires et d'utiliser correctement le mécanisme de synchronisation. Grâce à des moyens tels que l'exécution asynchrone, les verrous mutex et les variables de condition, nous pouvons efficacement empêcher les blocages. Dans le processus de développement actuel, nous devons pleinement comprendre les caractéristiques et les mécanismes de la programmation simultanée et concevoir raisonnablement le modèle de concurrence pour améliorer l'efficacité et la stabilité du programme.

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