Pergunta

Existe uma maneira de ver o que o aplicativo de um decorador Python fez com uma função à qual o apliquei.Por exemplo se eu tiver

class A(object):

   @property
   def something(self):
       return 0

Gostaria de ver para que serve o código executado something realmente parece.Existe uma maneira de fazer isso?

Foi útil?

Solução

Um decorador não produz código;um decorador é realmente apenas um açúcar sintático:

@property
def something(self):
    return 42

é realmente interpretado como:

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

por exemplo.a expressão que segue o @ o sinal é avaliado e o resultado é chamado, passando a função ou classe seguindo o @ linha.Qualquer que seja o decorador que retorne substitui o objeto original.

Para fins de introspecção, o @ linha é não retida;você teria que analisar o próprio código-fonte para descobrir quaisquer decoradores presentes.O decorador não é obrigado a devolver um objeto novo;você pode retornar o objeto original inalterado e não pode, com introspecção, saber a diferença.

Sua melhor aposta é retornar à fonte do decorador e apenas ler o código.O property decorador é implementado em C, mas o descritor como fazer contém uma implementação Python que faz a mesma coisa:

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 em: CC-BY-SA com atribuição
Não afiliado a StackOverflow
scroll top