Maison >développement back-end >Golang >Pourquoi mon code Go imprime-t-il la somme de 10 milliards de nombres au lieu de simplement « afficher le premier message : bonjour » ?

Pourquoi mon code Go imprime-t-il la somme de 10 milliards de nombres au lieu de simplement « afficher le premier message : bonjour » ?

DDD
DDDoriginal
2024-10-28 07:49:02288parcourir

Why does my Go code print the sum of 10 billion numbers instead of just

Concurrence Go et confusion des canaux

Problème

Un utilisateur essaie de comprendre la concurrence Go et les canaux en utilisant l'extrait de code suivant :

<code class="go">package main

import "fmt"

func display(msg string, c chan bool) {
    fmt.Println("display first message:", msg)
    c <- true
}

func sum(c chan bool) {
    sum := 0
    for i := 0; i < 10000000000; i++ {
        sum++
    }
    fmt.Println(sum)
    c <- true
}

func main() {
    c := make(chan bool)

    go display("hello", c)
    go sum(c)
    <-c
}</code>

Le résultat attendu est uniquement "afficher le premier message : bonjour" car la fonction principale doit se terminer une fois qu'elle reçoit des données du canal. Cependant, la sortie réelle inclut également la somme de 10 milliards de nombres.

Réponse

Le principal problème du code est que le planificateur peut choisir librement entre les deux goroutines (affichage et somme) qui ne sont pas bloqués. Alors que le programmeur s'attend à ce que l'affichage se termine en premier et envoie les données au canal avant la fin de la somme, cela peut ne pas se produire en raison de la nature non déterministe du planificateur.

Dans un scénario d'exécution possible :

  1. main crée deux goroutines pour l'affichage et la somme.
  2. Le planificateur passe immédiatement à l'affichage.
  3. l'affichage imprime son message et bloque l'attente qu'un récepteur accepte les données envoyées au canal .
  4. Le planificateur exécute la somme au lieu de reprendre l'affichage.
  5. sum calcule et imprime la somme de 10 milliards de nombres.
  6. Le planificateur choisit de reprendre l'affichage une fois la somme terminée.
  7. l'affichage envoie des données au canal.
  8. Le planificateur passe à main pour recevoir les données du canal.
  9. main imprime la somme et quitte le programme.

Pour résoudre ce problème et garantir que le message « Afficher le premier message : bonjour » est imprimé exclusivement, une approche consiste à utiliser un canal de résultat pour recevoir le message de l'affichage et terminer le programme immédiatement. La fonction principale modifiée serait :

<code class="go">func main() {
    result := make(chan string)

    go display("hello", result)
    fmt.Println(<-result)
}</code>

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