Table des matières
Contenu de la question
Solution de contournement
Solution
Maison développement back-end Golang Le programme Go se termine avant la fin du travail de goroutine

Le programme Go se termine avant la fin du travail de goroutine

Feb 08, 2024 pm 10:57 PM
go语言

Go 程序在 goroutine 工作完成之前退出

Dans cet article, l'éditeur php Xiaoxin présentera un problème important concernant les programmes Go : la situation de sortie avant que le travail de goroutine ne soit terminé. Dans le langage Go, goroutine est un thread léger qui peut exécuter des tâches simultanément. Cependant, lorsque notre programme peut se terminer avant la fin du travail goroutine, nous devons comprendre comment gérer cette situation pour garantir que notre programme peut terminer la tâche correctement. Dans le contenu suivant, nous explorerons ce problème et proposerons quelques solutions pour le résoudre.

Contenu de la question

J'ai du mal à comprendre comment bloquer et fermer correctement les chaînes. Je démarre un nombre arbitraire de travailleurs et je constate que ma fonction principale se termine avant la fin des travailleurs ou se bloque en raison de canaux non fermés. J'ai besoin d'un meilleur moyen d'empêcher le travailleur de lire le canal sans quitter le canal principal, puis de fermer gracieusement le canal une fois terminé pour mettre fin à la boucle. Toutes mes tentatives aboutissent à une impasse.

J'ai essayé plusieurs choses, notamment utiliser un groupe d'attente, mais le problème persiste. J'ai remarqué qu'en ajoutant time.sleep le programme fonctionne comme prévu, mais le commenter n'entraîne aucun travail.

time.sleep(time.duration(10 * time.second))
Copier après la connexion

Voici un exemple fonctionnel https://go.dev/play/p/qhqnj-ajqbi avec sleep préservé. Il s'agit du code cassé avec le délai d'attente de mise en veille commenté.

package main

import (
    "fmt"
    "sync"
    "time"
)

// some complicated work
func do(num int, ch chan<- int) {
    time.sleep(time.duration(500 * time.millisecond))
    ch <- num
}

func main() {

    results := make(chan int)

    // for some number of required complicated work
    for i := 0; i < 53; i++ {
        go do(i, results)
    }

    var wg sync.waitgroup

    // start 3 workers which can process results
    for i := 0; i < 3; i++ {
        wg.add(1)
        go func(id int) {
            defer wg.done()
            worker(id, results)
        }(i)
    }

    // handle closing the channel when all workers complete
    go func() {
        wg.wait()
        close(results)
    }()

    //time.sleep(time.duration(10 * time.second))

    fmt.println("donezo")
}

// process the results of do() in a meaningful way
func worker(id int, ch <-chan int) {
    fmt.println("starting worker", id)

    for i := range ch {
        fmt.println("channel val:", i)
    }
}
Copier après la connexion

J'ai aussi essayé de mettre defer wg.done() 移动到 worker() à l'intérieur de func mais c'est le même problème et ne fonctionne pas sans dormir.

// process the results of do() in a meaningful way
func worker(wg *sync.WaitGroup, id int, ch <-chan int) {
    fmt.Println("starting worker", id)

    defer wg.Done()

    for i := range ch {
        fmt.Println("channel val:", i)
    }
}
Copier après la connexion

Ai-je choisi le mauvais paradigme, ou est-ce que j'utilise simplement le mauvais paradigme ?

Solution de contournement

J'ai initialement demandé "Puis-je apporter quelques petits ajustements à mon code pour le faire fonctionner ? Ou dois-je repenser cela ? " La réponse que j'ai trouvée est, oui, il y a un petit ajustement.

J'ai dû apprendre un concept de base intéressant sur les canaux : on peut lire les données d'un canal fermé, c'est-à-dire vider le canal. Comme mentionné dans mon exemple d'origine, range ne se termine jamais car je ne trouve pas de bon endroit pour fermer la chaîne, et même lorsque je la force d'une autre manière créative, le programme présente un mauvais comportement

  • Quitter sans traiter tout le contenu de la chaîne
  • Interblocage ou envoi sur canal fermé

Cela est dû à une différence subtile dans le code "real", où le temps nécessaire pour traiter le contenu de la chaîne est plus long que le temps nécessaire pour remplir la chaîne et les choses sont désynchronisées.

Comme il n'y a pas de moyen pratique et clair dans mon expéditeur pour fermer la chaîne (ce qui est recommandé dans 99 % des didacticiels de chaîne), lorsque plusieurs travailleurs lisent la chaîne et que les travailleurs ne le savent pas, par goroutine dans C'est en fait Il est acceptable de le faire dans main où la dernière valeur est lue.

Solution

J'ai enveloppé le travailleur dans son propre sync.waitgroup et j'ai utilisé worker.wait() pour sync.waitgroup 中,并使用 worker.wait()阻止程序退出,从而允许工作“完成” ”。当没有更多数据要发送时,我独立地 close()

empêcher

le programme de se fermer, permettant ainsi au travail de "se terminer". ". Lorsqu'il n'y a plus de données à envoyer, je ferme() les canaux indépendamment, c'est à dire que je bloque en attendant que l'écrivain ait fini d'utiliser son propre groupe d'attente. close fournit une terminaison pour le cas des boucles de plage , car lorsque la valeur par défaut du canal est renvoyée, c'est-à-dire que le type eof est renvoyé lorsque la fin du canal est atteinte, il cessera de bloquer le canal d'intersection jusqu'à sa fermeture . workers.wait()Mon point de vue est que si vous ne savez pas combien de valeurs seront poussées en parallèle, go n'a aucun moyen de connaître la longueur du canal sans tampon car il est dans la portée,

jusqu'à ce que vous le fermiez.

. Puisqu'il est fermé, cela signifie lire tout ce qui reste jusqu'à la valeur de terminaison ou la fin. bloquera jusqu'à ce qu'il soit terminé.

Exemples d'opérations résolues

https://www.php.cn/link/2bf0ccdbb4d3ebbcb990af74bd78c658

Exemple de lecture d'une chaîne fermée

https://www.php.cn/link/d5397f1497b5cdaad7253fdc92db610b

🎜 🎜Sortie🎜
filling 0
filling 1
filling 2
filling 3
filling 4
filling 5
filling 6
filling 7
filling 8
filling 9
closed
empyting 0
empyting 1
empyting 2
empyting 3
empyting 4
empyting 5
empyting 6
empyting 7
empyting 8
empyting 9
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!

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

Outils d'IA chauds

Undresser.AI Undress

Undresser.AI Undress

Application basée sur l'IA pour créer des photos de nu réalistes

AI Clothes Remover

AI Clothes Remover

Outil d'IA en ligne pour supprimer les vêtements des photos.

Undress AI Tool

Undress AI Tool

Images de déshabillage gratuites

Clothoff.io

Clothoff.io

Dissolvant de vêtements AI

AI Hentai Generator

AI Hentai Generator

Générez AI Hentai gratuitement.

Article chaud

R.E.P.O. Crystals d'énergie expliqués et ce qu'ils font (cristal jaune)
2 Il y a quelques semaines By 尊渡假赌尊渡假赌尊渡假赌
Repo: Comment relancer ses coéquipiers
1 Il y a quelques mois By 尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island Adventure: Comment obtenir des graines géantes
4 Il y a quelques semaines By 尊渡假赌尊渡假赌尊渡假赌
Combien de temps faut-il pour battre Split Fiction?
3 Il y a quelques semaines By DDD

Outils chauds

Bloc-notes++7.3.1

Bloc-notes++7.3.1

Éditeur de code facile à utiliser et gratuit

SublimeText3 version chinoise

SublimeText3 version chinoise

Version chinoise, très simple à utiliser

Envoyer Studio 13.0.1

Envoyer Studio 13.0.1

Puissant environnement de développement intégré PHP

Dreamweaver CS6

Dreamweaver CS6

Outils de développement Web visuel

SublimeText3 version Mac

SublimeText3 version Mac

Logiciel d'édition de code au niveau de Dieu (SublimeText3)

Quel est le problème avec le fil de file d'attente dans GO's Crawler Colly? Quel est le problème avec le fil de file d'attente dans GO's Crawler Colly? Apr 02, 2025 pm 02:09 PM

Problème de threading de file d'attente dans Go Crawler Colly explore le problème de l'utilisation de la bibliothèque Crawler Crawler dans le langage Go, les développeurs rencontrent souvent des problèmes avec les threads et les files d'attente de demande. � ...

Quelles bibliothèques sont utilisées pour les opérations du numéro de point flottantes en Go? Quelles bibliothèques sont utilisées pour les opérations du numéro de point flottantes en Go? Apr 02, 2025 pm 02:06 PM

La bibliothèque utilisée pour le fonctionnement du numéro de point flottante dans le langage go présente comment s'assurer que la précision est ...

Comment résoudre le problème que les étiquettes de structure personnalisées à Goland ne prennent pas effet? Comment résoudre le problème que les étiquettes de structure personnalisées à Goland ne prennent pas effet? Apr 02, 2025 pm 12:51 PM

En ce qui concerne le problème des balises de structure personnalisées dans Goland lorsque vous utilisez Goland pour le développement du langage GO, vous rencontrez souvent des problèmes de configuration. L'un d'eux est ...

Quelles bibliothèques de GO sont développées par de grandes entreprises ou fournies par des projets open source bien connus? Quelles bibliothèques de GO sont développées par de grandes entreprises ou fournies par des projets open source bien connus? Apr 02, 2025 pm 04:12 PM

Quelles bibliothèques de GO sont développées par de grandes entreprises ou des projets open source bien connus? Lors de la programmation en Go, les développeurs rencontrent souvent des besoins communs, ...

Pourquoi est-il nécessaire de passer des pointeurs lors de l'utilisation de bibliothèques Go et Viper? Pourquoi est-il nécessaire de passer des pointeurs lors de l'utilisation de bibliothèques Go et Viper? Apr 02, 2025 pm 04:00 PM

GO POINTER SYNTAXE ET ATTENDRE DES PROBLÈMES DANS LA BIBLIOTHÈQUE VIPER Lors de la programmation en langage Go, il est crucial de comprendre la syntaxe et l'utilisation des pointeurs, en particulier dans ...

Dans Go, pourquoi les chaînes d'impression avec println et string () ont-elles des effets différents? Dans Go, pourquoi les chaînes d'impression avec println et string () ont-elles des effets différents? Apr 02, 2025 pm 02:03 PM

La différence entre l'impression de chaîne dans le langage go: la différence dans l'effet de l'utilisation de fonctions println et string () est en Go ...

Le langage GO est inefficace dans le traitement de l'accès à URL massif, comment l'optimiser? Le langage GO est inefficace dans le traitement de l'accès à URL massif, comment l'optimiser? Apr 02, 2025 am 10:15 AM

Stratégie d'optimisation des performances pour l'accès à URL massif du langage GO Cet article propose une solution d'optimisation des performances pour le problème de l'utilisation du langage GO pour traiter l'accès massif de l'URL. Programmes existants de CSV ...

Comment implémenter des opérations sur les listes liées Linux Iptables à Golang? Comment implémenter des opérations sur les listes liées Linux Iptables à Golang? Apr 02, 2025 am 10:18 AM

Utilisation de Golang pour implémenter Linux ...

See all articles