расширить Python namedtuple с помощью многих @properties?

Как namedtuples может быть расширен или разделен на подклассы со многими дополнительными @properties?
Для некоторых можно просто записать текст ниже; но существуют многие, таким образом, я ищу фабрика свойства или генератор. Один путь состоял бы в том, чтобы генерировать текст от _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
11
задан denis 3 February 2010 в 16:24
поделиться

3 ответа

Ответ на ваш вопрос

Как могут быть расширенный или подкласс с дополнительными @properties ?

: именно так, как вы это делаете! Какую ошибку вы получаете? Чтобы увидеть более простой случай,

>>> class x(collections.namedtuple('y', 'a b c')):
...   @property
...   def d(self): return 23
... 
>>> a=x(1, 2, 3)
>>> a.d
23
>>> 
18
ответ дан 3 December 2019 в 06:21
поделиться

Как насчет этого?

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)
2
ответ дан 3 December 2019 в 06:21
поделиться

Вот один подход, небольшой язык: превратите это в текст 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
-1
ответ дан 3 December 2019 в 06:21
поделиться
Другие вопросы по тегам:

Похожие вопросы: