Question

Existe-t-il un moyen de voir ce que l'application d'un décorateur Python a fait avec une fonction à laquelle je l'ai appliqué.Par exemple si j'ai

class A(object):

   @property
   def something(self):
       return 0

J'aimerais voir à quoi sert le code exécuté something ressemble en fait.Y a-t-il un moyen de faire cela?

Était-ce utile?

La solution

Un décorateur ne produit pas de code ;un décorateur n'est en réalité que du sucre syntaxique :

@property
def something(self):
    return 42

est en réalité interprété comme :

def something(self):
    return 42
something = property(something)

par exemple.l'expression qui suit le @ le signe est évalué et le résultat est appelé, en passant la fonction ou la classe qui suit le @ doubler.Quoi que le décorateur revienne ensuite remplace l'objet d'origine.

À des fins d'introspection, le @ la ligne est pas conservé;vous devrez analyser le code source lui-même pour découvrir les décorateurs présents.Un décorateur n’est pas obligé de restituer un nouvel objet ;vous pouvez restituer l'objet d'origine inchangé et vous ne pouvez pas, avec introspection, connaître la différence.

Le mieux est alors de revenir à la source du décorateur et de simplement lire le code.Le property decorator est implémenté en C, mais le descripteur comment faire contient une implémentation Python qui fait la même chose :

class Property(object):
    "Emulate PyProperty_Type() in Objects/descrobject.c"

    def __init__(self, fget=None, fset=None, fdel=None, doc=None):
        self.fget = fget
        self.fset = fset
        self.fdel = fdel
        if doc is None and fget is not None:
            doc = fget.__doc__
        self.__doc__ = doc

    def __get__(self, obj, objtype=None):
        if obj is None:
            return self
        if self.fget is None:
            raise AttributeError("unreadable attribute")
        return self.fget(obj)

    def __set__(self, obj, value):
        if self.fset is None:
            raise AttributeError("can't set attribute")
        self.fset(obj, value)

    def __delete__(self, obj):
        if self.fdel is None:
            raise AttributeError("can't delete attribute")
        self.fdel(obj)

    def getter(self, fget):
        return type(self)(fget, self.fset, self.fdel, self.__doc__)

    def setter(self, fset):
        return type(self)(self.fget, fset, self.fdel, self.__doc__)

    def deleter(self, fdel):
        return type(self)(self.fget, self.fset, fdel, self.__doc__)
Licencié sous: CC-BY-SA avec attribution
Non affilié à StackOverflow
scroll top