Quand j'ai commencé à programmer avec Python, si je ne me trompe pas, la version était la 3.3. Par conséquent, lorsque j'ai commencé à programmer, les décorateurs étaient disponibles depuis longtemps pour la communauté Python.
Les décorateurs de fonctions sont arrivés sur Python avec la version 2.2 et les décorateurs de classes sont arrivés sur Python avec la version 2.6.
Personnellement, je considère la fonctionnalité Decorator de Python comme une fonctionnalité très puissante du langage.
En fait, mon objectif est de faire une série d'articles sur les sujets les plus difficiles à comprendre en Python. Je compte aborder ces sujets, qui sont un peu plus d'une dizaine, un par un.
Dans cet article, j'essaierai d'aborder autant que possible chaque partie du sujet des décorateurs.
1. Contexte historique
- Premiers jours (pré-Python 2.2) : Avant les décorateurs, la modification de fonctions ou de classes impliquait souvent un emballage manuel ou un patch singe, ce qui était fastidieux et moins lisible.
- Métaclasses (Python 2.2) : Les métaclasses fournissaient un moyen de contrôler la création de classes, offrant certaines des fonctionnalités que les décorateurs fourniraient plus tard, mais elles étaient complexes pour des modifications simples.
- PEP 318 (Python 2.4) : Les décorateurs ont été formellement introduits dans Python 2.4 via PEP 318. La proposition s'inspire des annotations en Java et visait à fournir une manière plus propre et plus déclarative de modifier les fonctions et les méthodes. .
- Décorateurs de classe (Python 2.6) : Python 2.6 a étendu la prise en charge des décorateurs aux classes, améliorant encore leur polyvalence.
- Adoption généralisée : Les décorateurs sont rapidement devenus une fonctionnalité populaire, largement utilisée dans des frameworks comme Flask et Django pour le routage, l'authentification, etc.
2. Que sont les décorateurs ?
Essentiellement, un décorateur est un modèle de conception en Python qui vous permet de modifier le comportement d'une fonction ou d'une classe sans changer sa structure de base. Les décorateurs sont une forme de métaprogrammation, dans laquelle vous écrivez essentiellement du code qui manipule d'autres codes.
Vous savez que Python résout les noms en utilisant la portée indiquée dans l'ordre ci-dessous :
- Local
- Enfermant
- Mondial
- Intégré
Les décorateurs sont assis dans la portée Enclosing, qui est étroitement liée au concept de Closure.
Idée clé : Un décorateur prend une fonction en entrée, y ajoute des fonctionnalités et renvoie une fonction modifiée.
Analogie : Pensez à un décorateur comme à un emballage cadeau. Vous avez un cadeau (la fonction originale) et vous l'enveloppez avec du papier décoratif (le décorateur) pour le rendre plus joli ou ajouter des fonctionnalités supplémentaires (comme un nœud ou une carte). Le cadeau à l'intérieur reste le même, mais sa présentation ou les actions associées sont valorisées.
A) Variations du décorateur : basées sur la fonction ou basées sur la classe
La plupart des décorateurs en Python sont implémentés à l'aide de fonctions, mais vous pouvez également créer des décorateurs à l'aide de classes.
Les décorateurs basés sur les fonctions sont plus courants et plus simples, tandis que les décorateurs basés sur les classes offrent une flexibilité supplémentaire.
Syntaxe du décorateur de base basée sur les fonctions
def my_decorator(func): def wrapper(*args, **kwargs): # Do something before calling the decorated function print("Before function call") result = func(*args, **kwargs) # Do something after calling the decorated function print("After function call") return result return wrapper @my_decorator def say_hello(name): print(f"Hello, {name}!") say_hello("World")
Explication :
- my_decorator est la fonction décorateur. Il faut que la fonction func soit décorée en entrée.
- wrapper est une fonction interne qui encapsule l'appel de la fonction d'origine. Il peut exécuter du code avant et après la fonction d'origine.
- @my_decorator est la syntaxe du décorateur. C'est équivalent à say_hello = my_decorator(say_hello).
Syntaxe du décorateur de base basée sur les classes
Ceux-ci utilisent des classes au lieu de fonctions pour définir les décorateurs.
class MyDecorator: def __init__(self, func): self.func = func def __call__(self, *args, **kwargs): # Do something before calling the decorated function print("Before function call") result = self.func(*args, **kwargs) # Do something after calling the decorated function print("After function call") return result @MyDecorator def say_hello(name): print(f"Hello, {name}!") say_hello("World")
Explication :
- MyDecorator est une classe qui fait office de décorateur.
- La méthode __init__ stocke la fonction à décorer.
- La méthode __call__ rend l'instance de classe appelable, lui permettant d'être utilisée comme une fonction.
B) Mise en œuvre d'un décorateur simple
Le concept fondamental des décorateurs est que ce sont des fonctions qui prennent une autre fonction comme argument et étendent son comportement sans la modifier explicitement.
Voici le formulaire le plus simple :
def my_decorator(func): def wrapper(): print("Something is happening before the function is called.") func() print("Something is happening after the function is called.") return wrapper # Using the decorator with @ syntax @my_decorator def say_hello(): print("Hello!") # When we call say_hello() say_hello() # This is equivalent to: # say_hello = my_decorator(say_hello)
C) Implémentation d'un décorateur avec des arguments
Créons un décorateur qui enregistre le temps d'exécution d'une fonction :
def decorator_with_args(func): def wrapper(*args, **kwargs): # Accept any number of arguments print(f"Arguments received: {args}, {kwargs}") return func(*args, **kwargs) # Pass arguments to the original function return wrapper @decorator_with_args def greet(name, greeting="Hello"): print(f"{greeting}, {name}!") greet("Alice", greeting="Hi") # Prints arguments then "Hi, Alice!"
D) Implémentation d'un décorateur paramétré
Ce sont des décorateurs qui peuvent accepter leurs propres paramètres :
def repeat(times): def decorator(func): def wrapper(*args, **kwargs): for _ in range(times): result = func(*args, **kwargs) return result return wrapper return decorator @repeat(times=3) def greet(name): print(f"Hello {name}") return "Done" greet("Bob") # Prints "Hello Bob" three times
E) Implémentation d'un décorateur de classe
def singleton(cls): instances = {} def get_instance(*args, **kwargs): if cls not in instances: instances[cls] = cls(*args, **kwargs) return instances[cls] return get_instance @singleton class DatabaseConnection: def __init__(self): print("Initializing database connection") # Creating multiple instances actually returns the same instance db1 = DatabaseConnection() # Prints initialization db2 = DatabaseConnection() # No initialization printed print(db1 is db2) # True
F) Implémentation des décorateurs de méthodes
Ceux-ci sont spécialement conçus pour les méthodes de classe :
def debug_method(func): def wrapper(self, *args, **kwargs): print(f"Calling method {func.__name__} of {self.__class__.__name__}") return func(self, *args, **kwargs) return wrapper class MyClass: @debug_method def my_method(self, x, y): return x + y obj = MyClass() print(obj.my_method(5, 3))
G) Implémentation du chaînage de décorateurs
Plusieurs décorateurs peuvent être appliqués à une seule fonction :
def bold(func): def wrapper(): return "<b>" + func() + "</b>" return wrapper def italic(func): def wrapper(): return "<i>" + func() + "</i>" return wrapper @bold @italic def greet(): return "Hello!" print(greet()) # Outputs: <b><i>Hello!</i></b>
Explication :
- Les décorateurs sont appliqués de bas en haut.
- C'est plutôt dans ce qu'on fait en maths : f(g(x)).
- l'italique est appliqué en premier, puis le gras.
H) Que se passe-t-il si nous n'utilisons pas @functools.wraps ?
Le décorateur functools.wraps, Voir la documentation, est une fonction d'assistance qui préserve les métadonnées de la fonction d'origine (comme son nom, sa docstring et sa signature) lorsque vous l'enveloppez avec un décorateur. Si vous ne l'utilisez pas, vous perdrez ces informations importantes.
Exemple :
def my_decorator(func): def wrapper(*args, **kwargs): """Wrapper docstring""" return func(*args, **kwargs) return wrapper @my_decorator def my_function(): """My function docstring""" pass print(my_function.__name__) print(my_function.__doc__)
Sortie :
wrapper Wrapper docstring
Problème :
- Le nom de la fonction d'origine (my_function) et la docstring ("Ma fonction docstring") sont perdus.
- Cela peut rendre le débogage et l'introspection difficiles.
Solution : utilisez functools.wraps) :
def my_decorator(func): def wrapper(*args, **kwargs): # Do something before calling the decorated function print("Before function call") result = func(*args, **kwargs) # Do something after calling the decorated function print("After function call") return result return wrapper @my_decorator def say_hello(name): print(f"Hello, {name}!") say_hello("World")
Sortie :
class MyDecorator: def __init__(self, func): self.func = func def __call__(self, *args, **kwargs): # Do something before calling the decorated function print("Before function call") result = self.func(*args, **kwargs) # Do something after calling the decorated function print("After function call") return result @MyDecorator def say_hello(name): print(f"Hello, {name}!") say_hello("World")
Avantages de functools.wraps :
- Préserve les métadonnées de la fonction.
- Améliore la lisibilité et la maintenabilité du code.
- Facilite le débogage.
- Aide avec les outils d'introspection et les générateurs de documentation.
I) Décorateurs avec Etat
Les décorateurs peuvent également conserver l’état entre les appels de fonction. Ceci est particulièrement utile pour des scénarios tels que la mise en cache ou le comptage des appels de fonction.
def my_decorator(func): def wrapper(): print("Something is happening before the function is called.") func() print("Something is happening after the function is called.") return wrapper # Using the decorator with @ syntax @my_decorator def say_hello(): print("Hello!") # When we call say_hello() say_hello() # This is equivalent to: # say_hello = my_decorator(say_hello)
Sortie :
def decorator_with_args(func): def wrapper(*args, **kwargs): # Accept any number of arguments print(f"Arguments received: {args}, {kwargs}") return func(*args, **kwargs) # Pass arguments to the original function return wrapper @decorator_with_args def greet(name, greeting="Hello"): print(f"{greeting}, {name}!") greet("Alice", greeting="Hi") # Prints arguments then "Hi, Alice!"
Explication :
La fonction wrapper maintient un compteur (appels) qui s'incrémente à chaque fois que la fonction décorée est appelée.
Ceci est un exemple simple de la façon dont les décorateurs peuvent être utilisés pour maintenir l'état.
J) Meilleures pratiques pour les décorateurs Python
- Utilisez functools.wraps : Utilisez toujours @functools.wraps dans vos décorateurs pour préserver les métadonnées de la fonction d'origine.
- Gardez les décorateurs simples : Les décorateurs devraient idéalement faire une chose spécifique et bien la faire. Cela les rend plus réutilisables et plus faciles à comprendre.
- Documentez vos décorateurs : Expliquez ce que fait votre décorateur, quels arguments il prend et ce qu'il renvoie.
- Testez vos décorateurs : Écrivez des tests unitaires pour vous assurer que vos décorateurs fonctionnent comme prévu dans divers scénarios.
- Considérez l'ordre d'enchaînement : Soyez attentif à l'ordre lorsque vous enchaînez plusieurs décorateurs, car il affecte le flux d'exécution.
K) Mauvaises implémentations (anti-modèles)
- Décorateurs trop complexes : Évitez de créer des décorateurs trop complexes ou d'essayer de faire trop de choses. Cela les rend difficiles à comprendre, à maintenir et à déboguer.
- Ignorer functools.wraps : Oublier d'utiliser @functools.wraps entraîne une perte des métadonnées de fonction, ce qui peut entraîner des problèmes d'introspection et de débogage.
- Effets secondaires : Les décorateurs ne devraient idéalement pas avoir d'effets secondaires involontaires en dehors de la modification de la fonction décorée.
- Valeurs de codage en dur : Évitez les valeurs de codage en dur au sein des décorateurs. Utilisez plutôt des usines de décorateurs pour les rendre configurables.
- Ne pas gérer correctement les arguments : Assurez-vous que votre fonction wrapper peut gérer un nombre illimité d'arguments de position et de mots-clés en utilisant *args et **kwargs si le décorateur est destiné à être utilisé avec une variété de fonctions.
L) 10. Cas d'utilisation réels
- Journalisation : Enregistrement des appels de fonction, des arguments et des valeurs de retour pour le débogage ou l'audit.
- Timing : Mesure du temps d'exécution des fonctions pour l'analyse des performances.
- Mise en cache : Stockage des résultats d'appels de fonctions coûteux pour éviter les calculs redondants (mémorisation).
- Authentification et autorisation : Vérifier les informations d'identification ou les autorisations de l'utilisateur avant d'exécuter une fonction.
- Validation des entrées : Vérifier si les arguments passés à une fonction répondent à certains critères.
- Limitation du taux : Contrôler le nombre de fois qu'une fonction peut être appelée au cours d'une période de temps spécifique.
- Réessayer logique : Réessayer automatiquement un appel de fonction s'il échoue en raison d'une erreur temporaire.
- Tâches spécifiques au framework : Les frameworks comme Flask et Django utilisent des décorateurs pour le routage (mapping des URL aux fonctions), l'enregistrement des plugins, et bien plus encore.
M) Listes organisées de décorateurs Python
Vous pouvez trouver ci-dessous une liste organisée de décorateurs Python :
- Superbes décorateurs Python
- Bibliothèque de décorateurs Python
N) 11. Conclusion
Les décorateurs sont une fonctionnalité puissante et élégante de Python qui vous permet d'améliorer les fonctions et les classes de manière propre et déclarative.
En comprenant les principes, les meilleures pratiques et les pièges potentiels, vous pouvez exploiter efficacement les décorateurs pour écrire un code plus modulaire, maintenable et expressif.
Ils constituent un outil précieux dans l'arsenal de tout programmeur Python, en particulier lorsqu'il travaille avec des frameworks ou crée des composants réutilisables.
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!

Python et C ont chacun leurs propres avantages, et le choix doit être basé sur les exigences du projet. 1) Python convient au développement rapide et au traitement des données en raison de sa syntaxe concise et de son typage dynamique. 2) C convient à des performances élevées et à une programmation système en raison de son typage statique et de sa gestion de la mémoire manuelle.

Le choix de Python ou C dépend des exigences du projet: 1) Si vous avez besoin de développement rapide, de traitement des données et de conception du prototype, choisissez Python; 2) Si vous avez besoin de performances élevées, de faible latence et de contrôle matériel, choisissez C.

En investissant 2 heures d'apprentissage Python chaque jour, vous pouvez améliorer efficacement vos compétences en programmation. 1. Apprenez de nouvelles connaissances: lire des documents ou regarder des tutoriels. 2. Pratique: Écrivez du code et complétez les exercices. 3. Revue: consolider le contenu que vous avez appris. 4. Pratique du projet: Appliquez ce que vous avez appris dans les projets réels. Un tel plan d'apprentissage structuré peut vous aider à maîtriser systématiquement Python et à atteindre des objectifs de carrière.

Les méthodes pour apprendre Python efficacement dans les deux heures incluent: 1. Passez en revue les connaissances de base et assurez-vous que vous connaissez l'installation de Python et la syntaxe de base; 2. Comprendre les concepts de base de Python, tels que les variables, les listes, les fonctions, etc.; 3. Master Basic et Advanced Utilisation en utilisant des exemples; 4. Apprenez des erreurs courantes et des techniques de débogage; 5. Appliquer l'optimisation des performances et les meilleures pratiques, telles que l'utilisation des compréhensions de la liste et le suivi du guide de style PEP8.

Python convient aux débutants et à la science des données, et C convient à la programmation système et au développement de jeux. 1. Python est simple et facile à utiliser, adapté à la science des données et au développement Web. 2.C fournit des performances et un contrôle élevés, adaptés au développement de jeux et à la programmation système. Le choix doit être basé sur les besoins du projet et les intérêts personnels.

Python est plus adapté à la science des données et au développement rapide, tandis que C est plus adapté aux performances élevées et à la programmation système. 1. La syntaxe Python est concise et facile à apprendre, adaptée au traitement des données et à l'informatique scientifique. 2.C a une syntaxe complexe mais d'excellentes performances et est souvent utilisée dans le développement de jeux et la programmation système.

Il est possible d'investir deux heures par jour pour apprendre Python. 1. Apprenez de nouvelles connaissances: apprenez de nouveaux concepts en une heure, comme les listes et les dictionnaires. 2. Pratique et pratique: utilisez une heure pour effectuer des exercices de programmation, tels que la rédaction de petits programmes. Grâce à une planification et à une persévérance raisonnables, vous pouvez maîtriser les concepts de base de Python en peu de temps.

Python est plus facile à apprendre et à utiliser, tandis que C est plus puissant mais complexe. 1. La syntaxe Python est concise et adaptée aux débutants. Le typage dynamique et la gestion automatique de la mémoire le rendent facile à utiliser, mais peuvent entraîner des erreurs d'exécution. 2.C fournit des fonctionnalités de contrôle de bas niveau et avancées, adaptées aux applications haute performance, mais a un seuil d'apprentissage élevé et nécessite une gestion manuelle de la mémoire et de la sécurité.


Outils d'IA chauds

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

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

Undress AI Tool
Images de déshabillage gratuites

Clothoff.io
Dissolvant de vêtements AI

Video Face Swap
Échangez les visages dans n'importe quelle vidéo sans effort grâce à notre outil d'échange de visage AI entièrement gratuit !

Article chaud

Outils chauds

Version crackée d'EditPlus en chinois
Petite taille, coloration syntaxique, ne prend pas en charge la fonction d'invite de code

ZendStudio 13.5.1 Mac
Puissant environnement de développement intégré PHP

DVWA
Damn Vulnerable Web App (DVWA) est une application Web PHP/MySQL très vulnérable. Ses principaux objectifs sont d'aider les professionnels de la sécurité à tester leurs compétences et leurs outils dans un environnement juridique, d'aider les développeurs Web à mieux comprendre le processus de sécurisation des applications Web et d'aider les enseignants/étudiants à enseigner/apprendre dans un environnement de classe. Application Web sécurité. L'objectif de DVWA est de mettre en pratique certaines des vulnérabilités Web les plus courantes via une interface simple et directe, avec différents degrés de difficulté. Veuillez noter que ce logiciel

MantisBT
Mantis est un outil Web de suivi des défauts facile à déployer, conçu pour faciliter le suivi des défauts des produits. Cela nécessite PHP, MySQL et un serveur Web. Découvrez nos services de démonstration et d'hébergement.

mPDF
mPDF est une bibliothèque PHP qui peut générer des fichiers PDF à partir de HTML encodé en UTF-8. L'auteur original, Ian Back, a écrit mPDF pour générer des fichiers PDF « à la volée » depuis son site Web et gérer différentes langues. Il est plus lent et produit des fichiers plus volumineux lors de l'utilisation de polices Unicode que les scripts originaux comme HTML2FPDF, mais prend en charge les styles CSS, etc. et présente de nombreuses améliorations. Prend en charge presque toutes les langues, y compris RTL (arabe et hébreu) et CJK (chinois, japonais et coréen). Prend en charge les éléments imbriqués au niveau du bloc (tels que P, DIV),