Chapitre 6 : Closures et Générateurs
"My spelling is Wobbly. It’s good spelling but it Wobbles, and the letters get in the wrong places." — Winnie-the-Pooh
En plongée
Ayant grandit en fils de bibliothécaire et de spécialiste en langue anglaise, j'ai toujours été fasciné par les langages. Pas les langages de programmation. Enfin si, les langages de programmation, mais aussi les langages naturels. Prenez l'anglais. L'anglais est un langage schizophrénique qui emprunte des mots de l'allemand, du français, de l’espagnol et du latin (pour en citer quelque uns). En fait, "emprunte" n'est pas le bon mot ; "pille" est plus approprié. Ou peut être, "assimile" — comme les Borgs. Ouais, j'aime ça.
Nous sommes les Borgs. Vos caractèristiques linguistiques et étymologiques seront ajoutées aux nôtres. La résistance est futile.
Dans ce chapitre, vous allez apprendre les noms pluriels. Mais aussi, les fonctions qui retournent d'autres fonctions, les expressions régulières avancées, et les générateurs. Mais d'abord, discutons de comment former les noms pluriels. (Si vous n'avez pas encore lu le chapitre sur les expressions régulières, maintenant serait le bon moment. Ce chapitre assume que vous comprenez les bases des expressions régulières, et il en vient rapidement à des usages plus avancés.)
Si vous avez grandit dans un pays anglophone ou apprit l'anglais dans un contexte scolaire, vous êtes probablement familier avec les règles de base :
Si un mot se termine avec un S, X, ou Z, ajoutez ES. Bass devient basses, fax devient faxes, et waltz devient waltzes.
Si un mot se termine avec un H sonore, ajoutez ES ; s'il se termine avec un H muet, ajoutez juste un S. Qu'est-ce qu'un H sonore ? C'est un H qui combiné avec d'autres lettres forme un son que vous pouvez entendre. Donc coach devient coaches et rash devient rashes, parce que vous pouvez entendre les sons CH et SH quand vous les prononcez. Mais cheetah devient cheetahs, parce que le H est muet.
Si un mot se termine avec un Y qui sonne comme in I, remplacez le Y par IES ; si le Y est combiné avec une voyelle pour sonner différemment, ajoutez juste un S. Donc vacancy devient vacancies, mais day devient days.
Si tout échoue, alors ajoutez simplement S et espérez que ça ira.
(Je sais, il y a plein d'exceptions. Man devient men et woman devient women, mais human devient humans. Mouse devient mice and louse devient lice, mais house devient houses. Knife devient knives et wife devient wives, mais lowlife devient lowlifes. Et ne parlons même pas des mots qui sont leur propre pluriel comme sheep, deer et haiku.)
Bien évidemment, les autres langues sont complètement différentes.
Développons une librairie Python qui forme automatiquement le pluriel des mots anglais. Nous commencerons seulement avec ces quatre règles, mais gardez à l'esprit que vous allez inévitablement en ajouter d'autres.
Je sais, utilisons les expression régulières !
Dons vous recherchez des mots qui, au moins en anglais, signifie que vous recherchez des chaînes de caractères. Vous avez des règles qui disent que vous devez trouver différentes combinaisons de caractères, et leur faire différentes choses. Cela semble être un boulot pour les expressions régulières !
(1) C'est une expression régulière, mais elle utilise une syntaxe que vous n'avez pas vu dans le chapitre Expressions Régulières. Les crochets signifient "correspond exactement à un de ces caractères". Donc [sxz]
signifie " s
, ou x
, ou z
", mais seulement à un d'entre eux. Le $
devrait être familier ; il correspond à la fin de chaîne. Combinés, cette expression régulière vérifie si noun
se termine avec s
, x
ou z
.
(2) Cette fonction re.sub()
exécute une substitution de chaînes par expression régulière.
Regardons les substitutions par expression régulière plus en détail.
(1) Est-ce que la chaîne Mark
contient a
, b
, ou c
? Oui, elle contient a
.
(2) OK, maintenant trouvons a
, b
, ou c
, et replaçons-le par o
. Mark
devient Mork
.
(3) La même fonction transforme rock
en rook
.
(4) Vous devez penser que cela devrait transformer caps
en oaps
, mais en fait non. re.sub
remplace toutes les correspondances, pas seulement la première. Donc cette expression régulière transforme caps
en oops
, parce que le c
et le a
sont tous les deux transformés en o
.
Et maintenant, revenons à la fonction plural()
…
(1) Ici, vous remplacez la fin de chaîne (correspondant à $
) par la chaîne es
. En d'autres mots, cela ajoute es
à la chaîne. Vous pouvez obtenir la même chose avec la concaténation de chaîne, par exemple noun + 'es'
, mais j'ai choisi d'utiliser les expressions régulières pour chaque règle, pour des raisons qui deviendront plus claires plus loin dans ce chapitre.
(2) Regardez attentivement, c'est une autre nouvelle variation. Le ^
, en premier caractère entre crochets, signifie quelque chose de spécial : la négation. [^abc]
signifie "n'importe quel caractère sauf a
, b
, ou c
". Donc [^aeioudgkprt]
signifie n'importe quel caractère sauf a
, e
, i
, o
, u
, d
, g
, k
, p
, r
, ou t
. Puis ce caractère doit être suivi par h
, suivi par la fin de chaîne. Vous cherchez les mots qui finissent avec un H où le H est sonore.
(3) Même motif ici : trouver les mots qui se terminent par Y, où le caractère avant le Y n'est pas a
, e
, i
, o
, ou u
. Vous cherchez les mots qui finissent avec un Y qui sonne comme un I.
Regardons la négation d'expression régulière plus en détail.
(1) vacancy
correspond à cette expression régulière, parce qu'il se termine en cy
, et c
n'est pas a
, e
, i
, o
, ou u
.
(2) boy
ne correspond pas, parce qu'il se termine en oy
, et vous avez spécifiquement dit que le caractère avant le y
ne peut pas être o
. day
ne correspond pas, parce qu'il se termine en ay
.
(3) pita
ne correspond pas, parce qu'il ne se termine pas en y
.
(1) Cette expression régulière transforme vacancy
en vacancies
et agency
en agencies
, ce qui est ce que vous voulez. Notez que cela transforme aussi boy
en boies
, mais cela n'arrivera jamais dans la fonction parce que vous avez fait cette recherche d'abord avec re.search
pour savoir si vous devez la faire avec re.sub
.
(2) Juste en passant, je voudrais pointer le fait qu'il est possible de combiner ces deux expressions régulières (une pour trouver si la règle s'applique, et une autre pour effectivement l'appliquer) en une unique expression régulière. Cela ressemblerait à ça. Cela doit sembler en grande partie familier : vous utilisez un groupe de capture, que vous avez appris dans l’Étude de Cas : Analyser les Numéros de Téléphone. Le groupe est utilisé pour capturer le caractère avant la lettre y
. Puis dans la chaîne de substitution, vous utilisez une nouvelle syntaxe, \1
, qui signifie « hé, tu te souviens de ce premier groupe ? met le ici. » Dans ce cas, vous capturez le c
avant le y
; quand vous faite la substitution, vous substituez c
à la place de c
, et ies
à la place de y
. (Si vous avez plus d'un groupe de capture, vous pouvez utiliser \2
et \3
et ainsi de suite.)
Les substitutions par expression régulière sont extrêmement puissantes, et la syntaxe \1
les rend encore plus puissante. Mais combiner toute l’opération en une seule expression régulière est aussi plus difficile à lire, et cela ne correspond pas directement à la façon dont vous décrivez les règles du pluriel. Vous avez initialement posé les règles comme "si le mot se termine en S, X, ou Z, alors ajouter ES". Si vous regardez cette fonction, vous avez deux lignes de code qui disent "si le mot se termine en S, X, ou Z, alors ajouter ES". Il n'est pas possible d'être plus direct.
Une liste de fonctions
Maintenant, vous allez ajouter un niveau d'abstraction. Vous avez commencé par définir une liste de règles : si ceci, faire cela, autrement aller à la prochaine règle. Compliquons temporairement une partie du programme, ainsi vous pourrez en simplifier une autre partie.
(1) Maintenant, chaque règle de correspondance est sa propre fonction qui retourne les résultats de l'appel de la fonction re.search()
.
(2) Chaque règle d'application est aussi sa propre fonction qui appelle la fonction re.sub()
pour appliquer la règle de pluriel appropriée.
(3) Au lieu d'avoir une fonction (plural()
) avec de multiple règles, vous avez la structure de données rules
qui est une séquence de paires de fonctions.
(4) Puisque les règles ont été placées dans une structure de données séparée, la nouvelle fonction plural()
peut être réduite à quelques lignes de code. En utilisant une boucle for
, vous pouvez extraire les règles de correspondance et les règles d'application deux à la fois (une de correspondance et une d'application) de la structure rules
. À la première itération de la boucle for
, matches_rule
recevra match_sxz
, et apply_rule
recevra apply_sxz
. À la seconde itération (en présumant que vous arrivez jusque là), matches_rules
sera assignée à match_h
, et apply_rule
sera assignée à apply_h
. La fonction est garantie de retourner quelque chose éventuellement, parce que la dernière règle (match_default
) retourne simplement True
, cela signifie que la règle d'application correspondante (apply_default
) sera toujours appliquée.
La raison pour laquelle cette technique fonctionne est que tout en Python est un objet, y compris les fonctions. La structure de données rules
contient des fonctions — pas des noms de fonctions, mais de vrais objets fonction. Quand ils sont assignés dans la boucle for
, alors matches_rule
et apply_rule
sont de vraies fonctions que vous pouvez appeler. À la première itération de la boucle for
, c'est équivalent à appeler matches_sxz(noun)
, et si elle retourne une correspondance, à appeler apply_sxz(noun)
.
Si cet additionnel niveau d'abstraction prête à confusion, essayez de dérouler la fonction pour voir l’équivalence. La boucle for
complète est équivalente à :
Le bénéfice ici est que la fonction plural()
est maintenant simplifiée. Elle prend une séquence de règles, définies quelque part, et les itére d'une manière générique.
Trouver une règle de correspondance.
Une correspondance ? Alors appeler la règle d'application et retourner le résultat.
Pas de correspondance ? Aller a l’étape 1.
Les règles peuvent être définies n'importe où, de n'importe quelle façon. La fonction plural()
ne s'en préoccupe pas.
Maintenant, est-ce qu'ajouter ce niveau d'abstraction en valait la peine ? Et bien, pas encore. Considérez ce qu'il faudrait pour ajouter une nouvelle règle à la fonction. Dans le premier exemple, cela nécessiterait d'ajouter une instruction if
à la fonction plural()
. Dans le second exemple, cela nécessiterait d'ajouter deux fonctions, match_foo()
et apply_foo()
, et puis mettre à jour la séquence rules
pour spécifier où dans la séquence les nouvelles fonctions de correspondance et d'application doivent être appelées par rapport aux autres règles.
Mais ce n'est qu'une étape dans la prochaine section. Allons-y…
Une liste de motifs
Définir des fonctions nommées séparées pour chaque règle de correspondance et d'application n'est pas vraiment nécessaire. Vous ne les appelez jamais directement ; vous les ajoutez à la séquence rules
et les appelez depuis-là. De plus, chaque fonction suit l’un des deux motifs. Toutes les fonctions de correspondance appelent re.search()
et toutes les fonctions d'applications appelent re.sub()
. Examinons les motifs afin de définir de nouvelles règles plus faciles :
(1) build_match_and_apply_functions()
est une fonction qui construit d'autres fonction de manière dynamique. Elle prend pattern
, search
et replace
, puis définit une fonction matches_rule()
qui appelle re.search()
avec le pattern
(motif) qui a été passé à la fonction build_match_and_apply_functions()
, et le word
(mot) qui a été passé à la fonction matches_rule()
que vous construisez. Waouh.
(2) Construire la fonction d'application fonctionne de la même manière. La fonction d'application est une fonction qui prend un paramètre, et qui appelle re.sub()
avec les paramètres search
et replace
qui ont été passés à la fonction build_match_and_apply_functions()
, et (le mot) word
qui a été passé à la fonction apply_rule()
que vous construisez. Cette technique utilisant les valeurs de paramètres externes au sein d'une fonction dynamique est appelée closures. Vous définissez essentiellement des constantes au sein de la fonction d'application que vous construisez : elle prend un paramètre (word
), mais elle le traite alors avec deux autres valeurs (search
et replace
) qui ont été paramétrées quand vous avez défini la fonction application.
(3) Finalement, la fonction build_match_and_apply_functions()
retourne un tuple de deux valeurs : les deux fonctions que vous venez de créer. Les constantes que vous avez définies dans ces fonctions (pattern
dans la fonction matches_rule()
, et search
dans la fonction apply_rule()
) restent avec ces fonctions, même après que votre retour de build_match_and_apply_functions()
. C'est incroyablement cool.
Si cela prête vraiment à confusion (et cela devrait, c'est bizarre), cela devrait devenir plus clair en regardant comment l'utiliser.
(1) Nos "règles" du pluriel sont maintenant définies comme tuple de tuples de chaînes (non pas de fonctions). La première chaîne dans chaque groupe est le motif d'expression régulière que vous utiliseriez avec re.search()
pour voir si la règle correspond. La deuxième et la troisième chaîne dans chaque groupe sont les expressions de recherche et de remplacement que vous utiliseriez avec re.sub()
pour effectivement appliquer la règle afin de transformer un nom en son pluriel.
(2) Il y a un léger changement ici, dans la règle par défaut. Dans l'exemple précédent, la fonction match_default()
retournait simplement True
, signifiant que si aucune des règles spécifiques ne correspond, le code ajouterait simplement un s
à la fin du mot donné. Cette exemple fait quelque chose de fonctionnellement équivalent. La dernière expression régulière demande si le mot a une fin ($
correspond à la fin d'une chaîne). Bien sûr, toute chaîne a une fin, même une chaîne vide, donc cette expression correspond toujours. Ainsi, elle sert le même objectif que la fonction matches_default()
qui retournait toujours True
: elle assure que si aucune règle spécifique ne correspond, le code ajoute un s
à la fin du mot donné.
(3) Cette ligne est magique. Elle prend la séquence de chaînes dans patterns
et les transforme en une séquence de fonctions. Comment ? En faisant correspondre les chaînes à la fonction build_match_and_apply_functions()
. C'est-à-dire, elle prend chaque triplet de chaînes et appelle la fonction build_match_and_apply_functions()
avec ces trois chaînes comme arguments. La fonction build_match_and_apply_functions()
retourne un tuple de deux fonctions. Cela signifie que rules
est donc fonctionnellement équivalente à l'exemple précédent : une liste de tuples où chaque tuple est une paire de fonctions. La première fonction est la fonction de correspondance qui appelle re.search()
, et la seconde fonction est la fonction d'application qui appelle re.sub()
.
Cette version du script est complétée par le point d’entrée principal, la fonction plural()
.
(1) Puisque la liste rules
est la même que dans l'exemple précédent (c'est vraiment le cas), cela ne devrait pas être surprenant que la fonction plural()
n'a pas changé du tout. Elle est complètement générique ; elle prend une liste de fonctions de règle et les appelle dans l'ordre. Elle ne se soucie pas de comment les règles sont définies. Dans l'exemple précédent, elles étaient définies comme des fonctions nommées séparées. Maintenant, elles sont construites dynamiquement en faisant correspondre la sortie de la fonction build_match_and_apply_functions()
avec une liste brute de chaînes. Cela n'a aucune d'importance ; la fonction plural()
fonctionne toujours de la même façon.
Un fichier de motifs
Vous avez extrait tout le code dupliqué et ajoutez assez d'abstraction pour les que les règles du pluriel soient définies dans une liste de chaînes. La prochaine étape logique est de prendre ces chaînes et de les mettre dans un fichier séparé, où elles peuvent être maintenues séparément du code qui les utilise.
En premier, créons un fichier texte qui contient les règles que vous voulez. Pas de structure de données sophistiquée, juste des chaînes délimitées par des espaces dans trois colonnes. Appelons-le plural4-rules.txt
.
Maintenant regardons comment nous pouvons utiliser ce fichier de règles.
(1) La fonction build_match_and_apply_functions()
n'a pas changé. Vous utilisez toujours des closures pour construire deux fonctions dynamiquement qui utilisent les variables définies dans la fonction externe.
(2) La fonction globale open()
ouvre un fichier et retourne un objet fichier. Dans ce cas, le fichier que nous ouvrons contient les motifs de chaînes pour mettre au pluriel des noms. L'instruction with
crée ce qui est appelé un contexte : quand le bloc with
se termine, Python va automatiquement fermer le fichier, même si une exception se produit à l’intérieur du bloc with
. Vous en apprendrez plus à-propos des blocs with
et des objets fichier dans le chapitre Fichiers.
(3) L'idiome for line in <fileobject>
lit les données depuis le fichier ouvert, une ligne à la fois, et assigne le texte à la variable line
. Vous en apprendrez plus à-propos de la lecture de fichier dans le chapitre Fichiers.
(4) Chaque ligne du fichier a en fait trois valeurs, mais elles sont séparées par des espaces blancs (tabulation ou espace, cela n'a pas d'importance). Pour les séparer, utilisez la méthode de chaîne split()
. Le premier argument de la méthode split()
est None
, ce qui signifie "découpe à chaque espace blanc (tabulation ou espace, cela n'a pas d'importance)." Le second argument est 3
, ce qui signifie "découpe sur les espaces blancs 3 fois, puis laisse en l’état le reste de la ligne." Une ligne comme [sxz]$ $ es
sera découpée en liste ['[sxz]$', '$', 'es']
, ce qui signifie que pattern
aura '[sxz]$
', search
aura '$
', et replace
aura 'es
'. C'est beaucoup de puissance dans une seule petite ligne de code.
(5) Finalement, vous passez pattern
, search
, et replace
à la fonction build_match_and_apply_functions()
, qui retourne un tuple de fonctions. Vous ajouter ce tuple à la liste rules
, et rules
finit par stocker la liste des fonctions de correspondance et d'application que la fonction plural()
attend.
L’amélioration ici est que vous avez complètement séparé les règles du pluriel dans une fichier externe, ainsi il peut être maintenu séparément du code qui l'utilise. Le code avec le code, les données avec les données, et tout va bien.
Générateurs
Est-ce que ce ne serait pas grandiose d'avoir une fonction générique plural()
qui analyse le fichier de règles ? Récupère les règles, cherche une correspondance, applique la transformation appropriée, va à la règle suivante. C'est tout ce que la fonction plural()
a à faire, et c'est tout ce que la fonction plural()
devrait faire.
Mais comment est-ce que ça fonctionne ? Regardons d'abord un premier exemple interactif.
(1) La présence du mot-clef yield
dans make_counter
signifie que ce n'est pas une fonction normale. C'est un type spécial de fonction qui génère des valeurs l'une après l'autre. Vous pouvez l'imaginez comme une fonction pouvant être interrompue et reprise. L'appeler retournera un générateur qui peut être utiliser pour générer des valeurs successives de x
.
(2) Pour créer une instance du générateur make_counter
, appelez-le juste comme une autre fonction. Notez qu'en fait cela n’exécute pas le code de la fonction. Vous pouvez le voir parce que la première ligne de la fonction make_counter
appelle print()
, mais rien n'est encore affiché.
(3) La fonction make_counter
retourne un objet générateur.
(4) La fonction next()
prend un objet générateur et retourne sa prochaine valeur. La première fois que vous appelez next()
avec le générateur counter
, elle exécute le code dans make_counter
jusqu'à la première instruction yield
, puis retourne la valeur qui a été générée. Dans ce cas, ce sera 2
, car vous avez initialement créé le générateur en appelant make_counter(2)
.
(5) Appeler de façon répétitive next()
avec le mème objet générateur reprend exactement là où il s'était arrêté et continue jusqu'à ce qu'il rencontre la prochaine instruction yield
. Toutes les variables, état local, etc., sont sauvegardées à yield
et rétablies à next()
. La prochaine ligne de code attendant d’être exécutée appelle print()
, qui affiche incrementing x
. Après cela, l'instruction x = x + 1
. Puis elle boucle à nouveau au travers de la boucle while
, et la première chose recontrée est l'instruction yield x
, qui sauvegarde l’état de toute chose et retourne la valeur courante de x
(maintenant 3
).
(6) La deuxième fois que vous appelez next()
, vous faites à nouveau les même choses, mais cette fois x
vaut 4
.
Puisque make_counter
définit une boucle infinie, vous pouvez théoriquement faire cela pour toujours, et ça continuerait juste d’incrémenter x
et de cracher des valeurs. Mais regardons plutôt des utilisations plus productives des générateurs.
“yield” met en pause une fonction. “next()” la reprend où elle s’était interrompue.
Un générateur Fibonacci
(1) La séquence de Fibonacci est une séquence de nombres où chaque nombre est la somme des deux nombres qui le précèdent. Elle commence avec 0
et 1
, progresse doucement d'abord, puis de plus en plus rapidement. Pour commencer la séquence, vous avez besoin de deux variables : a
commence à 0
, et b
commence à 1
.
(2) Générons a
qui est le nombre courant dans la séquence.
(3) Assignons b
, qui est le prochain nombre dans la séquence, à a
, mais calculons aussi la prochaine valeur (a + b)
et assignons-la à b
pour l'utiliser plus tard. Notez que cela se produit en parallèle ; si a
vaut 3
et b
vaut 5
, alors a, b = b, a + b
paramètrera 5
à a
(la précédente valeur de b
) et 8
à b
(la somme des précédentes valeurs de a
et b
).
Donc vous avez une fonction qui restitue les nombres Fibonacci successifs. Bien sûr, vous pouvez faire cela avec la récursion, mais cette manière est plus facile à lire. De même, cela fonctionne bien avec les boucles for
.
(1) Vous pouvez utiliser un générateur tel que fib()
directement dans la boucle for
. La boucle for
appellera automatiquement la fonction next()
, récupèrera les valeurs du générateur fib()
et les assignera à la variable d'index (n
) de la boucle for
.
(2) A chaque passage dans la boucle for
, n
reçoit une nouvelle valeur de l'instruction yield
dans fib()
, tout ce que vous avez à faire est de l'afficher. Une fois que fib()
est à court de nombres (a
devient plus grand que max
, qui dans ce cas vaut 1000
), alors la boucle for
se termine normalement.
(3) C'est un idiome utile : passez un générateur à la fonction list()
, et il itérera le générateur tout entier (exactement comme la boucle for
de l'exemple précédent) et retournera une liste de toutes les valeurs.
Un générateur de règles du pluriel
Retournons à plural5.py
et voyons comment cette version de plural()
fonctionne.
(1) Ici, pas de magie. Souvenez-vous que les lignes du fichier de règles ont trois valeurs séparées par des espaces, donc vous utilisez line.split(None, 3)
pour récupérer les trois "colonnes" et les assigner à trois variables locales.
(2) Et puis vous générez. Qu'est-ce que vous générez ? Deux fonctions, construites dynamiquement avec notre vieille amie, build_match_and_apply_functions()
, qui est identique à celle des exemples précédents. En d'autres mots, rules()
est un générateur qui crache des fonctions de correspondance et d'application à la demande.
(3) Puisque rules()
est un générateur, vous pouvez l'utiliser directement dans une boucle for
. Au premier passage dans la boucle for
, vous appellerez la fonction rules()
, qui ouvrira le fichier de motifs, lira la première ligne, construira dynamiquement une fonction de correspondance et une fonction d'application à partir des motifs de cette ligne, et générera les fonctions construites dynamiquement. Au second passage dans la boucle for
, vous reprendrez exactement là où vous avez laissé rules()
(qui était au milieu de la boucle for line in pattern_file
). La première chose qu'elle fera sera de lire la ligne suivante du fichier (qui est toujours ouvert), de construire dynamiquement une autre fonction de correspondance et une autre fonction d'application sur la base des motifs de cette ligne, et générera les deux fonctions.
Qu'avez-vous gagné par rapport à l’étape 4 ? Le temps de démarrage. À l’étape 4, quand vous importez le module plural4
, il lit tout le fichier de motifs et construit une liste de toutes les règles possibles, avant mème que ne vous pensiez à appeler la fonction plural()
. Avec les générateurs, vous pouvez tout faire de façon paresseuse : vous lisez la première règle et créez les fonctions puis les testez, et si ça fonctionne vous ne lisez même pas le reste du fichier ou créez d'autres fonctions.
Qu'avez-vous perdu ? De la performance ! Chaque fois que vous appelez la fonction plural()
, le générateur rules
recommence depuis le début — ce qui signifie ouvrir à nouveau le fichier de motifs et le lire depuis le début, une ligne à la fois.
Et si vous pouviez avoir le meilleur des deux mondes : temps de démarrage minimal (ne pas exécuter de code à import
), et performance maximale (ne pas construire les même fonctions encore et encore). Oh, et vous voulez toujours garder les règles dans un fichier séparé (parce que le code avec le code et les données avec les données), tant que vous n'avez jamais besoin de lire deux fois la même ligne.
Pour faire cela, vous allez devoir construire votre propre itérateur. Mais avant que vous fassiez cela, vous devez apprendre les classes Python.
Pour aller plus loin (en anglais)
Last updated