Итак, я играю с декораторами в Python 2.6, и у меня возникают некоторые проблемы с их работой. Вот мой файл класса:
class testDec:
@property
def x(self):
print 'called getter'
return self._x
@x.setter
def x(self, value):
print 'called setter'
self._x = value
Я думал, что это означает, что нужно обращаться x
со свойством, но вызывать эти функции при получении и установке. Итак, я запустил IDLE и проверил это:
>>> from testDec import testDec
from testDec import testDec
>>> t = testDec()
t = testDec()
>>> t.x
t.x
called getter
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "testDec.py", line 18, in x
return self._x
AttributeError: testDec instance has no attribute '_x'
>>> t.x = 5
t.x = 5
>>> t.x
t.x
5
Ясно, что первый вызов работает, как и ожидалось, так как я вызываю метод получения, и значения по умолчанию не существует, и он завершается ошибкой. ОК, хорошо, я понимаю. Тем не менее, вызов метода assign t.x = 5
создает новое свойство x
, и теперь метод get не работает!
Чего мне не хватает?
Ответы:
Похоже, вы используете классические классы старого стиля в Python 2. Чтобы свойства работали правильно, вам нужно использовать классы нового стиля (в Python 2 вы должны наследовать от
object
). Просто объявите свой класс какMyClass(object)
:Оно работает:
Другая деталь, которая может вызвать проблемы, заключается в том, что оба метода должны иметь одинаковое имя для работы свойства. Если вы определите сеттер с другим именем, как это, он не будет работать :
И еще одна вещь, которую поначалу не совсем легко обнаружить, это порядок: сначала нужно определить получатель . Если вы сначала определите установщик, вы получите
name 'x' is not defined
ошибку.источник
Просто примечание для других людей, которые спотыкаются здесь, ища это исключение: обе функции должны иметь одинаковое имя. Наименование методов следующим образом приведет к исключению:
Вместо этого дайте обоим методам одно и то же имя
Также важно отметить, что порядок декларации имеет значение. Получатель должен быть определен до установки в файле, иначе вы получите
NameError: name 'x' is not defined
источник
NameError: name 'x' is not defined
исключение.Вам нужно использовать классы нового стиля, которые вы делаете, получая свой класс из объекта:
Тогда это должно работать.
источник
В случае, если кто-то пришел сюда из Google, в дополнение к ответам выше, я хотел бы добавить, что это требует тщательного внимания при вызове
__init__
метода установки из метода вашего класса на основе этого ответа. В частности:источник
__init__
метода, но и из любого другого метода в классе.