Maison > développement back-end > Golang > Comment un blocage peut-il se produire lors de l'utilisation de groupes d'attente et de canaux tamponnés dans Go ?

Comment un blocage peut-il se produire lors de l'utilisation de groupes d'attente et de canaux tamponnés dans Go ?

Linda Hamilton
Libérer: 2024-10-28 04:16:30
original
729 Les gens l'ont consulté

How Can Deadlock Occur When Using WaitGroups and Buffered Channels in Go?

Détection des blocages dans la concurrence Go avec les groupes d'attente

Dans Go, la concurrence est souvent gérée à l'aide de canaux et de groupes d'attente pour orchestrer des goroutines. Cependant, il est essentiel de comprendre les pièges potentiels qui peuvent conduire à des blocages.

Description du problème

Considérez le code suivant qui tente d'utiliser des canaux et des groupes d'attente mis en mémoire tampon :

<code class="go">package main

import (
    "fmt"
    "sync"
)

func main() {
    ch := make(chan []int, 4)
    var m []int

    var wg sync.WaitGroup
    for i := 0; i < 5; i++ {
        wg.Add(1)
        go func() {
            defer wg.Done()
            ch <- m // Send to channel
            return
        }()
    }
    wg.Wait() // Wait for all goroutines to complete

    for c := range ch {
        fmt.Printf("c is %v", c) // Iterate over channel
    }
}</code>
Copier après la connexion

Bien que le canal s'attende à se fermer automatiquement une fois sa capacité atteinte, ce code entraîne de manière inattendue une erreur de blocage.

Solution

Il existe deux problèmes clés menant à l'impasse :

  1. Capacité de canal insuffisante : Le tampon de canal a une capacité de 4, alors qu'il y a 5 goroutines qui tentent d'y écrire. Il en résulte une situation dans laquelle les goroutines en attente d'écriture sont bloquées car le canal est plein.
  2. Range Over Unclosed Channel : La boucle pour c := range ch continue d'écouter les éléments entrants du chaîne indéfiniment, en attendant que la chaîne soit fermée. Cependant, comme il ne reste plus de goroutines pour écrire sur la chaîne, celle-ci ne se ferme jamais.

Pour sortir de l'impasse, deux solutions sont :

Solution 1 : Agrandissement de la capacité du canal et fermeture explicite

<code class="go">ch := make(chan []int, 5) // Increase channel capacity
...
wg.Wait()
close(ch) // Close the channel to stop range loop</code>
Copier après la connexion

Cela garantit qu'il y a suffisamment d'espace dans le canal et le ferme explicitement, permettant à la boucle de plage de se terminer.

Solution 2 : Signal Done Condition dans Goroutine

<code class="go">func main() {
    ch := make(chan []int, 4)
    var m []int

    var wg sync.WaitGroup
    for i := 0; i < 5; i++ {
        wg.Add(1)
        go func() {
            ch <- m
            wg.Done() // Signal completion within goroutine
            return
        }()
    }
    go func() {
        for c := range ch {
            fmt.Printf("c is %v\n", c)
            wg.Done() //Decrement count for each iteration
        }
    }()
    wg.Wait()
}</code>
Copier après la connexion

Dans cette solution, chaque goroutine signale son achèvement en appelant wg.Done() dans la goroutine elle-même. Le groupe d'attente est également décrémenté dans la boucle de plage pour chaque itération, garantissant que wg.Wait() se termine finalement et que le programme se termine.

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!

source:php.cn
Déclaration de ce site Web
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
Derniers articles par auteur
Tutoriels populaires
Plus>
Derniers téléchargements
Plus>
effets Web
Code source du site Web
Matériel du site Web
Modèle frontal