Maison  >  Article  >  développement back-end  >  Décorateurs de fonctions en Python : comprendre les méthodes @property, Getter et Setter

Décorateurs de fonctions en Python : comprendre les méthodes @property, Getter et Setter

DDD
DDDoriginal
2024-09-22 18:25:32113parcourir

Function Decorators in Python: Understanding @property, Getter, and Setter Methods

Dans la programmation orientée objet, l'encapsulation est un concept fondamental crucial pour garantir l'intégrité des données et cacher les détails d'implémentation à l'utilisateur. Python, connu pour sa simplicité et sa lisibilité, utilise des getters et des setters dans le cadre de cette encapsulation. Cet article examine l'objectif et la mise en œuvre des getters et setters en Python, fournissant un aperçu de leur rôle dans la gestion de l'accès aux données et le maintien de l'intégrité des objets. En particulier, nous explorerons comment le décorateur @property en Python simplifie ces concepts, permettant une approche plus pythonique de l'accès et de la mise à jour des attributs d'objet.

Encapsulation et importance des variables privées
Au cœur de l'encapsulation se trouve l'idée de masquage des données : contrôler l'accès à l'état interne d'un objet pour éviter toute interférence involontaire ou toute utilisation abusive. Cela nécessite l'utilisation de variables privées. Dans de nombreux langages de programmation, les variables privées sont utilisées pour garantir que les données sensibles d'un objet ne peuvent pas être consultées ou modifiées directement sans autorisation appropriée, ce qui préserve l'intégrité de l'objet donné.
Python n'a pas de variables privées strictes comme certains autres langages, mais utilise à la place une convention consistant à préfixer un attribut avec un trait de soulignement simple () ou double (_) pour indiquer qu'il est destiné à un usage interne. Décomposons la différence entre ces deux conventions.

Single Underscore (_) vs. Double Underscore (__) en Python

a. Souligné unique (_) :

  • Un seul trait de soulignement au début d'une variable (par exemple, _price) est une convention utilisée pour indiquer que l'attribut est destiné à un usage interne. Il n’est pas strictement appliqué par Python, ce qui signifie que l’attribut est toujours accessible depuis l’extérieur de la classe (c’est-à-dire qu’il n’est pas privé). Cependant, cela signale aux autres développeurs que l'attribut est "protégé" et ne doit pas être accédé directement sauf si cela est nécessaire. Exemple:
class Product:
    def __init__(self, price):
        self._price = price  # Protected attribute (convention)

product = Product(10)
print(product._price)  # Accessing is possible, but discouraged

b. Double trait de soulignement (__) :

  • Un double trait de soulignement au début d'une variable (par exemple, __price) déclenche une modification du nom. La modification du nom modifie le nom de l'attribut en interne pour empêcher tout accès ou modification accidentel depuis l'extérieur de la classe. Cela rend l'attribut plus difficile d'accès direct même s'il n'est toujours pas complètement privé : Python renomme l'attribut en interne en le préfixant avec _ClassName, le rendant accessible uniquement par son nom mutilé (par exemple, _Product__price). Exemple:
class Product:
    def __init__(self, price):
        self.__price = price  # Name-mangled attribute

product = Product(10)
# print(product.__price)  # This will raise an AttributeError
print(product._Product__price)  # Accessing the mangled attribute
  • Ils sont utiles lorsque vous souhaitez éviter le remplacement accidentel d'attributs dans des sous-classes ou si vous souhaitez une protection plus renforcée contre un accès externe involontaire.

Pourquoi utiliser des attributs privés ?
Les attributs privés, en particulier ceux indiqués par un seul trait de soulignement (_), sont importants pour maintenir l'encapsulation. Ils protègent l’état interne d’un objet en décourageant le code externe d’interagir directement avec lui, ce qui aide :

  1. Préserver l'intégrité des données : les attributs privés empêchent la modification accidentelle des données internes sensibles ou critiques.
  2. Activer l'accès contrôlé : en utilisant les méthodes getter et setter (ou le décorateur @property), l'objet contrôle comment et quand ses attributs sont accessibles ou modifiés, en ajoutant souvent une logique de validation.
  3. Améliorez la maintenabilité : les détails internes étant masqués, vous pouvez modifier l'implémentation sous-jacente sans affecter le comportement externe de votre classe.

Méthodes traditionnelles Getter et Setter
Dans de nombreux langages de programmation, les getters et setters sont utilisés pour fournir un accès contrôlé aux variables privées. Voir l'exemple ci-dessous :

class Product:
    def __init__(self, price):
        self._price = price  # Protected attribute

    def get_price(self):
        return self._price

    def set_price(self, value):
        if value >= 0:
            self._price = value
        else:
            raise ValueError("Price cannot be negative")

product = Product(10)
print(product.get_price())  # 10
product.set_price(20)
print(product.get_price())  # 20

Dans cet exemple, le getter (get_price()) et le setter (set_price()) fournissent un moyen d'accéder et de modifier l'attribut _price tout en appliquant certaines règles (comme s'assurer que le prix n'est pas négatif).

Le décorateur @property
Python offre une manière plus élégante de gérer l'accès aux attributs privés à l'aide du décorateur @property. Ce décorateur vous permet de définir des méthodes qui se comportent comme des attributs, rendant le code plus lisible et pythonique tout en permettant un accès contrôlé.

Utilisation du décorateur @property pour Getter et Setter
Vous trouverez ci-dessous l'exemple précédent refactorisé avec @property pour simplifier la syntaxe et améliorer la lisibilité :

class Product:
    def __init__(self, price):
        self._price = price

    @property
    def price(self):
        return self._price

    @price.setter
    def price(self, value):
        if value >= 0:
            self._price = value
        else:
            raise ValueError("Price cannot be negative")

product = Product(10)
print(product.price)  # 10
product.price = 20
print(product.price)  # 20

Dans cette version refactorisée :

  • Le décorateur @property nous permet d'accéder à price() comme un attribut, c'est-à-dire product.price, plutôt que d'avoir à appeler une méthode getter comme product.get_price().

  • Le décorateur @price.setter active la logique de définition de la valeur du prix, nous permettant de le définir comme product.price = 20 tout en appliquant les règles de validation.

Pourquoi utiliser @property ?
Le décorateur @property rend votre code plus propre et plus facile à utiliser, en particulier lorsqu'il s'agit d'attributs privés. Voici pourquoi :

  1. Lisibilité : il permet d'accéder naturellement aux attributs tout en gardant cachée la logique sous-jacente de validation ou de transformation.
  2. Encapsulation : vous pouvez appliquer des règles sur la façon dont les attributs sont accessibles ou modifiés sans exposer les détails d'implémentation internes.
  3. Flexibilité : vous pouvez refactoriser le comportement interne sans modifier l'interface externe, ce qui signifie que le reste de votre base de code ne sera pas affecté.

Conclusion
L'encapsulation est la pierre angulaire de la programmation orientée objet, et l'utilisation par Python de variables privées, ainsi que du décorateur @property, fournit un moyen propre et flexible de gérer l'accès à l'état interne d'un objet. Alors que les attributs avec un seul trait de soulignement (_) signalent qu'ils sont destinés à un usage interne, les attributs avec deux traits de soulignement (__) offrent une protection plus renforcée grâce à la modification du nom. Le décorateur @property vous permet d'implémenter un accès contrôlé à ces attributs privés de manière pythonique et lisible, garantissant l'intégrité des données tout en conservant une interface publique propre.

Références

  • Documents Python sur la propriété

  • PEP 318 : Décorateurs de fonctions

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