Изменение аргументов конструктора namedtuple с помощью подкласса?
Я хочу создать namedtuple
, который представляет отдельные флаги в коротком поле бит. Я пытаюсь подклассифицировать его так, чтобы я мог распаковать битполе до того, как будет создан кортеж. Однако моя текущая попытка не работает:
class Status(collections.namedtuple("Status", "started checking start_after_check checked error paused queued loaded")):
__slots__ = ()
def __new__(cls, status):
super(cls).__new__(cls, status & 1, status & 2, status & 4, status & 8, status & 16, status & 32, status & 64, status & 128)
Теперь мой опыт работы с super()
ограничен, и мой опыт работы с __new__
практически не существует, поэтому я не совсем уверен, что сделать (для меня) загадочную ошибку TypeError: super.__new__(Status): Status is not a subtype of super
. Поиск в Google и поиск в документах не дали ничего поучительного.
Справка
Ответы
Ответ 1
У вас почти было это:-) Есть только две небольшие исправления:
- Новый метод требует оператора return
- Супервызов должен иметь два аргумента: cls и Status
Полученный код выглядит следующим образом:
import collections
class Status(collections.namedtuple("Status", "started checking start_after_check checked error paused queued loaded")):
__slots__ = ()
def __new__(cls, status):
return super(cls, Status).__new__(cls, status & 1, status & 2, status & 4, status & 8, status & 16, status & 32, status & 64, status & 128)
Он работает чисто, как и ожидалось:
>>> print Status(47)
Status(started=1, checking=2, start_after_check=4, checked=8, error=0, paused=32, queued=0, loaded=0)
Ответ 2
Я бы избегал super
, если вы явно не пользуетесь множественным наследованием (надеюсь, здесь не так;-). Просто сделайте что-нибудь вроде::
def __new__(cls, status):
return cls.__bases__[0].__new__(cls,
status & 1, status & 2, status & 4,
status & 8, status & 16, status & 32,
status & 64, status & 128)