Maison > développement back-end > Golang > Comment la responsabilité de la fermeture des canaux dans les générateurs Go doit-elle être gérée ?

Comment la responsabilité de la fermeture des canaux dans les générateurs Go doit-elle être gérée ?

Mary-Kate Olsen
Libérer: 2024-12-08 11:56:17
original
326 Les gens l'ont consulté

How Should Responsibility for Closing Channels in Go Generators Be Handled?

Implémentation idiomatique de générateurs avec rendement

Dans Go, l'implémentation de générateurs utilisant le rendement est généralement réalisée grâce à une combinaison de goroutines et de canaux. La fonction génératrice crée une goroutine qui génère des valeurs via un canal, et la fonction consommateur reçoit ces valeurs du canal dans une boucle for-range.

Responsabilité de la fermeture du canal

Selon le Allez les idiomes, la responsabilité de la fermeture du canal incombe à la fonction générateur. Étant donné que le générateur sait quand l'itération est terminée, il doit fermer le canal pour signaler au consommateur qu'aucune autre valeur ne sera reçue.

Code modifié avec la fermeture différée de l'appelant()

Dans votre code modifié, vous avez correctement placé la responsabilité de la fermeture du canal sur l'appelant en ne le fermant pas dans la fonction générateur. Cependant, vous devez également supprimer l'appel close() dans la fonction main(), car il est incorrect de fermer un canal déjà fermé.

package main

import (
    "./lib"
    "fmt"
)

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

func main() {
    channel := lib.PermutateWithChannel(fruits)
    defer close(channel) // Defer closing the channel

    for myFruits := range channel {
        fmt.Println(myFruits)
        if myFruits[0] == banned {
            break // Break from the loop instead of closing the channel
        }
    }
}
Copier après la connexion

Effets secondaires négatifs de la fermeture d'un canal fermé

Effets secondaires négatifs de la fermeture d'un canal fermé

Lorsque l'appelant ferme le canal, toute tentative ultérieure de lui envoyer des valeurs entraînera une panique à l'exécution. En effet, le canal est marqué comme fermé et l'envoi vers un canal fermé est illégal. Cependant, cette panique n'a pas d'effets secondaires négatifs au-delà de la fin de la goroutine qui a tenté d'envoyer des valeurs.

Type de retour de canal de réception uniquement
type ReceiveOnlyChannel <-chan []string

func NewReceiveOnlyChannel(channel <-chan []string) *ReceiveOnlyChannel {
    return (*ReceiveOnlyChannel)(&channel)
}

func PermutateWithChannel(strings []string) *ReceiveOnlyChannel {
    // ... (same as before, except it returns ReceiveOnlyChannel)
}
Copier après la connexion

Pour restreindre le canal renvoyé par la fonction de bibliothèque pour recevoir -seulement, tout en permettant à l'appelant de le fermer, vous pouvez introduire un nouveau type qui enveloppe le canal et expose uniquement le canal de réception uniquement. canal :

En encapsulant le canal dans un nouveau type, vous pouvez restreindre son accessibilité pour recevoir uniquement des opérations tout en permettant à l'appelant de le fermer via la méthode Close() du type wrapper.

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
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