Puis-je voir ce qu’un décorateur Python fait à mon code ?
-
20-12-2019 - |
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?
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__)