Maison > développement back-end > Golang > Comment implémenter idiomatiquement des générateurs Go à l'aide de Goroutines et de canaux ?

Comment implémenter idiomatiquement des générateurs Go à l'aide de Goroutines et de canaux ?

Mary-Kate Olsen
Libérer: 2024-12-04 04:42:11
original
623 Les gens l'ont consulté

How to Idiomatically Implement Go Generators Using Goroutines and Channels?

La manière idiomatique d'implémenter des générateurs avec Yield dans Go pour les fonctions récursives

Dans Go, l'implémentation de générateurs utilisant le mot-clé Yield n'est pas directement prise en charge par la langue. Cependant, une approche idiomatique consiste à simuler cette fonctionnalité à l’aide de goroutines et de canaux. Voici un examen plus approfondi de vos questions :

1. Manière idiomatique d'implémenter des générateurs

La manière la plus idiomatique d'implémenter des générateurs avec le même comportement que le rendement est d'avoir une fonction basée sur une goroutine qui envoie des valeurs dans un canal. Cette goroutine devrait fermer le canal lorsque toutes les valeurs ont été générées. Voici un exemple :

package main

import (
    "fmt"
    "time"
)

// Generate integers in a goroutine.
func generateIntegers(numbers []int, c chan<- int) {
    defer close(c)

    for _, n := range numbers {
        c <- n
        time.Sleep(time.Second) // Simulate delay for clarity
    }
}

func main() {
    numbers := []int{1, 2, 3, 4, 5}
    c := make(chan int)

    go generateIntegers(numbers, c)

    // Consume the generated values.
    for v := range c {
        fmt.Println(v)
    }
}
Copier après la connexion

2. Responsabilité de la fermeture du canal

Idiomatiquement, la fonction génératrice (celle qui produit les valeurs) devrait être responsable de la fermeture du canal. Cela garantit que le canal est fermé lorsque toutes les valeurs ont été générées, permettant au consommateur de savoir quand arrêter de recevoir des valeurs.

3. Modification du code

La modification que vous proposez, pour rendre l'appelant responsable de la fermeture du canal, qu'il ait besoin ou non de toutes les permutations, est également une approche idiomatique. Cependant, il est important d'utiliser le report pour garantir que le canal est fermé même si une erreur se produit.

// Modify the caller code:

package main

import (
    "./lib"
    "fmt"
)

var (
    fruits  = []string{"apple", "banana", "cherry", "durian"}
    banned = "durian"
)

func main() {
    channel := lib.PermutateWithChannel(fruits)
    defer close(channel)

    for myFruits := range channel {
        fmt.Println(myFruits)
        if myFruits[0] == banned {
            // Close the channel when the banned fruit is encountered.
            // Since we're using defer, this will happen even if an error occurs.
            break
        }
    }
}
Copier après la connexion

4. Effets secondaires négatifs potentiels de la fermeture d'un canal après que l'appelant l'a fermé

Dans votre exemple, l'appelant ferme le canal une fois que le générateur a envoyé toutes les permutations. Cependant, la goroutine exécutant le code du générateur peut toujours essayer d'envoyer vers le canal fermé. Cela ne provoque aucun effet secondaire négatif observable, car la goroutine va simplement paniquer. Cependant, il est toujours recommandé de fermer les chaînes uniquement lorsqu'elles ne sont plus nécessaires pour éviter tout comportement inattendu et garantir une gestion efficace des ressources.

5. Renvoi d'un canal de réception uniquement

Il est possible de renvoyer un canal de réception uniquement à partir de la fonction générateur. Cela empêche l'appelant de fermer le canal, garantissant ainsi que seul le générateur en a la responsabilité. Voici comment procéder :

// Generator function returns a receive-only channel.
func generateIntegers(numbers []int) <-chan int {
    c := make(chan int)
    go func() {
        defer close(c)

        for _, n := range numbers {
            c <- n
            time.Sleep(time.Second) // Simulate delay for clarity
        }
    }()

    return c
}
Copier après la connexion

Cela garantit que l'appelant ne peut pas fermer le canal, évitant ainsi les problèmes décrits à la question 4.

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