Как правильно переопределить __setattr__ и __getattribute__ в классах нового стиля в Python?
Я хочу переопределить методы класса __getattribute__
и __setattr__
класса Python. Мой вариант использования является обычным: у меня есть несколько специальных имен, которые я хочу обработать, и я хочу, чтобы поведение по умолчанию было чем-то еще. Для __getattribute__
кажется, что я могу запросить поведение по умолчанию, просто подняв AttributeError
. Однако как я могу добиться того же в __setattr__
? Вот тривиальный пример, реализующий класс с неизменяемыми полями "A", "B" и "C".
class ABCImmutable(SomeSuperclass):
def __getattribute__(self, name):
if name in ("A", "B", "C"):
return "Immutable value of %s" % name
else:
# This should trigger the default behavior for any other
# attribute name.
raise AttributeError()
def __setattr__(self, name, value):
if name in ("A", "B", "C"):
raise AttributeError("%s is an immutable attribute.")
else:
# How do I request the default behavior?
???
Что происходит вместо вопросительных знаков? В классах старого стиля ответ был явно self.__dict__[name] = value
, но документация указывает, что это неправильно для классов нового стиля.
Ответы
Ответ 1
Это
super(ABCImmutable, self).__setattr__(name, value)
в Python 2 или
super().__setattr__(name, value)
в Python 3.
Кроме того, повышение AttributeError
не, как вы возвращаетесь к по умолчанию для __getattribute__
. Вы возвращаетесь к умолчанию по умолчанию с помощью
return super(ABCImmutable, self).__getattribute__(name)
на Python 2 или
return super().__getattribute__(name)
на Python 3.
Raising AttributeError
пропускает обработку по умолчанию и переходит в __getattr__
или просто создает AttributeError
в вызывающем коде, если нет __getattr__
.
См. документацию по Настройка доступа к атрибутам.
Ответ 2
SomeSuperclass.__setattr__(self, name, value)
?