recherche
Maisondéveloppement back-endTutoriel PythonDécorateurs Python : un guide complet

Python Decorators: A Comprehensive Guide

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 :

  1. Local
  2. Enfermant
  3. Mondial
  4. 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!

Déclaration
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
Python vs C: Comprendre les principales différencesPython vs C: Comprendre les principales différencesApr 21, 2025 am 12:18 AM

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.

Python vs C: Quelle langue choisir pour votre projet?Python vs C: Quelle langue choisir pour votre projet?Apr 21, 2025 am 12:17 AM

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.

Atteindre vos objectifs python: la puissance de 2 heures par jourAtteindre vos objectifs python: la puissance de 2 heures par jourApr 20, 2025 am 12:21 AM

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.

Maximiser 2 heures: stratégies d'apprentissage Python efficacesMaximiser 2 heures: stratégies d'apprentissage Python efficacesApr 20, 2025 am 12:20 AM

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.

Choisir entre Python et C: La bonne langue pour vousChoisir entre Python et C: La bonne langue pour vousApr 20, 2025 am 12:20 AM

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 vs C: Une analyse comparative des langages de programmationPython vs C: Une analyse comparative des langages de programmationApr 20, 2025 am 12:14 AM

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.

2 heures par jour: le potentiel de l'apprentissage python2 heures par jour: le potentiel de l'apprentissage pythonApr 20, 2025 am 12:14 AM

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 vs C: courbes d'apprentissage et facilité d'utilisationPython vs C: courbes d'apprentissage et facilité d'utilisationApr 19, 2025 am 12:20 AM

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

See all articles

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

Video Face Swap

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 !

Outils chauds

Version crackée d'EditPlus en chinois

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

ZendStudio 13.5.1 Mac

Puissant environnement de développement intégré PHP

DVWA

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

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

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