Maison > développement back-end > Golang > le corps du texte

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

DDD
Libérer: 2024-10-28 07:49:02
original
228 Les gens l'ont consulté

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>
Copier après la connexion

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>
Copier après la connexion

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
Tutoriels populaires
Plus>
Derniers téléchargements
Plus>
effets Web
Code source du site Web
Matériel du site Web
Modèle frontal