Maison >développement back-end >Golang >Pourquoi le code Go fourni avec WaitGroup et le canal mis en mémoire tampon entraîne-t-il un blocage ?
Deadlocks dans Go avec WaitGroup et Buffered Channels
Dans Go, un blocage se produit lorsque plusieurs goroutines s'attendent les unes les autres pour se terminer, ce qui entraîne dans une impasse. Cette situation peut survenir lors d'une utilisation incorrecte des canaux mis en mémoire tampon et des WaitGroups.
Considérez le code suivant :
<code class="go">package main import "fmt" import "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 return }() } wg.Wait() for c := range ch { fmt.Printf("c is %v", c) } }</code>
Ce code est censé créer un canal avec une taille de tampon de 4 et démarrer 5 goroutines. , chacun envoyant une tranche vide au canal. La goroutine principale attend que toutes les goroutines aient terminé, puis parcourt le canal.
Cependant, ce code entraînera une impasse. Pourquoi ?
Cause de l'impasse :
Deux problèmes existent dans le code :
Solutions :
Augmenter la capacité du canal : En augmentant la capacité du canal à 5, suffisamment d'emplacements seront disponibles pour que toutes les goroutines envoient leurs valeurs sans blocage. De plus, fermer le canal une fois que les goroutines ont fini d'écrire signalera à la boucle de portée qu'aucun autre élément n'arrive, l'empêchant d'attendre indéfiniment.
<code class="go">ch := make(chan []int, 5) ... wg.Wait() close(ch)</code>
Utiliser Terminé () dans la boucle : Au lieu de fermer le canal, on peut utiliser la méthode Done() de WaitGroup pour signaler quand la dernière goroutine est terminée. En appelant Done() dans la boucle de plage, la goroutine principale sera avertie lorsque le canal est vide et la boucle pourra se terminer.
<code class="go">go func() { for c := range ch { fmt.Printf("c is %v\n", c) wg.Done() } }() wg.Wait()</code>
Ces solutions résolvent l'impasse en garantissant que le canal a une capacité suffisante et que la boucle de plage se termine lorsqu'il n'y a plus d'éléments à lire dans le canal.
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!