Могу ли я увидеть, что декоратор Python делает с моим кодом?
-
20-12-2019 - |
Вопрос
Есть ли способ увидеть, что приложение декоратора Python сделало с функцией, к которой я его применил.Например, если у меня есть
class A(object):
@property
def something(self):
return 0
Я хотел бы посмотреть, для чего выполняется код something
на самом деле выглядит.Есть ли способ сделать это?
Решение
Декоратор не создает код;декоратор на самом деле представляет собой всего лишь синтаксический сахар:
@property
def something(self):
return 42
на самом деле интерпретируется как:
def something(self):
return 42
something = property(something)
напримервыражение, следующее за @
знак оценивается и вызывается результат, передавая функцию или класс после @
линия.Что бы потом ни возвращал декоратор заменяет исходный объект.
В целях самоанализа @
линия нет сохраняется;вам придется проанализировать сам исходный код, чтобы обнаружить любые присутствующие декораторы.Декоратор не обязан возвращать новый объект;вы можете вернуть исходный объект без изменений, но при самоанализе вы не сможете обнаружить разницу.
Лучше всего затем вернуться к исходному коду декоратора и просто прочитать код.А property
декоратор реализован на C, но дескриптор, инструкции содержит реализацию Python, которая делает то же самое:
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__)