Pregunta

¿Hay alguna manera de ver qué ha hecho la aplicación de un decorador de Python con una función a la que se lo he aplicado?Por ejemplo si tengo

class A(object):

   @property
   def something(self):
       return 0

Me gustaría ver para qué se ejecuta el código. something realmente parece.¿Hay alguna forma de hacer esto?

¿Fue útil?

Solución

Un decorador no produce código;un decorador es en realidad sólo azúcar sintáctico:

@property
def something(self):
    return 42

realmente se interpreta como:

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

p.ej.la expresión que sigue a la @ Se evalúa el signo y se llama al resultado, pasando la función o clase que sigue al @ línea.Lo que sea que luego devuelva el decorador. reemplaza el objeto original.

Para fines de introspección, el @ la línea es no retenido;Tendrías que analizar el código fuente para descubrir los decoradores presentes.Un decorador no está obligado a devolver un objeto nuevo;puedes devolver el objeto original inalterado y no puedes, con introspección, notar la diferencia.

Lo mejor que puede hacer es volver a la fuente del decorador y simplemente leer el código.El property decorador está implementado en C, pero el descriptor cómo contiene una implementación de Python que hace lo mismo:

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__)
Licenciado bajo: CC-BY-SA con atribución
No afiliado a StackOverflow
scroll top