Расширить Python namedtuple со многими @properties?
Как можно расширять или подклассифицировать namedtuples со многими дополнительными @properties?
Для некоторых можно просто написать текст ниже; но есть много,
поэтому я ищу генератор или свойство factory.
Один из способов - генерировать текст из _fields
и выполнять его;
другой - это add_fields с таким же эффектом во время выполнения.
(Мои @props должны получать строки и поля
в базе данных, разбросанной по нескольким таблицам,
так что rec.pname
persontable[rec.personid].pname
;
но namedtuples-with-smart-fields будут использовать и другие виды использования.)
""" extend namedtuple with many @properties ? """
from collections import namedtuple
Person = namedtuple( "Person", "pname paddr" ) # ...
persontable = [
Person( "Smith", "NY" ),
Person( "Jones", "IL" )
]
class Top( namedtuple( "Top_", "topid amount personid" )):
""" @property
.person -> persontable[personid]
.pname -> person.pname ...
"""
__slots__ = ()
@property
def person(self):
return persontable[self.personid]
# def add_fields( self, Top.person, Person._fields ) with the same effect as these ?
@property
def pname(self):
return self.person.pname
@property
def paddr(self):
return self.person.paddr
# ... many more
rec = Top( 0, 42, 1 )
print rec.person, rec.pname, rec.paddr
Ответы
Ответ 1
Ответ на ваш вопрос
Как продлить namedtuples или подклассы с дополнительным @properties
: именно так вы это делаете! Какую ошибку вы получаете? Чтобы увидеть более простой случай,
>>> class x(collections.namedtuple('y', 'a b c')):
... @property
... def d(self): return 23
...
>>> a=x(1, 2, 3)
>>> a.d
23
>>>
Ответ 2
Как насчет этого?
class Top( namedtuple( "Top_", "topid amount personid" )):
""" @property
.person -> persontable[personid]
.pname -> person.pname ...
"""
__slots__ = ()
@property
def person(self):
return persontable[self.personid]
def __getattr__(self,attr):
if attr in Person._fields:
return getattr(self.person, attr)
raise AttributeError("no such attribute '%s'" % attr)
Ответ 3
Здесь один подход, небольшой язык:
превратите это в текст Python, как указано выше, и выполните его.
(Расширение текста в текст легко сделать, и легко проверить —
вы можете посмотреть промежуточный текст.)
Я уверен, что есть похожие, если не такие-маленькие, ссылки, пожалуйста?
# example of a little language for describing multi-table databases 3feb
# why ?
# less clutter, toprec.pname -> persontable[toprec.personid].pname
# describe in one place: easier to understand, easier to change
Top:
topid amount personid
person: persontable[self.personid] + Person
# toprec.person = persontable[self.personid]
# pname = person.pname
# locid = person.locid
# todo: chaining, toprec.city -> toprec.person.loc.city
Person:
personid pname locid
loc: loctable[self.locid] + Loc
Loc:
locid zipcode province city