Maison > développement back-end > Golang > Pourquoi le passage d'un WaitGroup par valeur dans les canaux Go entraîne-t-il une impasse ?

Pourquoi le passage d'un WaitGroup par valeur dans les canaux Go entraîne-t-il une impasse ?

Mary-Kate Olsen
Libérer: 2024-10-31 04:59:02
original
513 Les gens l'ont consulté

Why does passing a WaitGroup by value in Go channels lead to a deadlock?

Impasse dans le canal Go en raison du passage de références

Dans Go, les canaux sont un mécanisme de communication puissant, mais ils peuvent conduire à des blocages sinon utilisé correctement. Une impasse se produit lorsque deux goroutines ou plus s'attendent l'une l'autre pour terminer une opération, ce qui empêche aucune des goroutines de continuer.

Considérez le programme Go suivant :

<br>package main</p>
<p>import (</p>
<div class="code" style="position:relative; padding:0px; margin:0px;"><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:php;toolbar:false">"fmt"
"sync"
Copier après la connexion
Copier après la connexion

)

func push(c chan int, wg sync.WaitGroup) {

for i := 0; i < 5; i++ {
    c <- i
}
wg.Done()
Copier après la connexion
Copier après la connexion

>

func pull(c chan int, wg sync.WaitGroup) {

for i := 0; i < 5; i++ {
    result, ok := <-c
    fmt.Println(result, ok)
}
wg.Done()
Copier après la connexion
Copier après la connexion

}

func main() {

var wg sync.WaitGroup
wg.Add(2)
c := make(chan int)

go push(c, wg)
go pull(c, wg)

wg.Wait()
Copier après la connexion

>

Lorsque vous exécutez ce programme, il paniquera avec une erreur de blocage :

throw: all goroutines are asleep - deadlock!
Copier après la connexion

Le problème réside dans la façon dont le WaitGroup est transmis aux goroutines . Dans Go, les structures sont transmises par valeur, ce qui signifie qu'une copie du WaitGroup est transmise à chaque goroutine. Lorsqu'une goroutine appelle Done sur sa copie, cela n'a aucun effet sur le WaitGroup d'origine créé dans la fonction principale.

Pour résoudre le blocage, le pointeur WaitGroup doit être passé à la place de la valeur. Voici le code corrigé :

<br>package main</p><p>import (</p><div class="code" style="position:relative; padding:0px; margin:0px;"><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:php;toolbar:false">"fmt"
"sync"
Copier après la connexion
Copier après la connexion

)

func push(c chan int , wg *sync.WaitGroup) {

for i := 0; i < 5; i++ {
    c <- i
}
wg.Done()
Copier après la connexion
Copier après la connexion

}

func pull(c chan int, wg *sync.WaitGroup) {

for i := 0; i < 5; i++ {
    result, ok := <-c
    fmt.Println(result, ok)
}
wg.Done()
Copier après la connexion
Copier après la connexion

}

func main() {

var wg sync.WaitGroup
wg.Add(2)
c := make(chan int)

go push(c, &wg)
go pull(c, &wg)

wg.Wait()
Copier après la connexion

}

En passant le pointeur WaitGroup au lieu de la valeur, les deux goroutines ont accès au même Instance WaitGroup. Lorsqu'une goroutine appelle Done, elle décrémente le décompte du WaitGroup d'origine, permettant à l'autre goroutine de continuer.

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
Article précédent:Comment pouvez-vous briser une boucle For en dehors de sa portée dans Golang ? Article suivant:Pourquoi les méthodes sur `T` ne sont-elles pas utilisables par `*T` dans Go ?
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
Derniers numéros
Rubriques connexes
Plus>
Recommandations populaires
Tutoriels populaires
Plus>
Derniers téléchargements
Plus>
effets Web
Code source du site Web
Matériel du site Web
Modèle frontal