如何在Python的namedtuple中添加多个@properties属性?

12

如何使用许多附加的@property 扩展或子类化namedtuple?
如果只有几个属性,可以直接编写下面的文本;但是如果有很多个, 我正在寻找一个生成器或属性工厂。 一种方法是从_fields生成文本并执行; 另一种方法是在运行时使用具有相同效果的add_fields。
(我的@props用于在散布在多个表中的数据库中获取行和字段, 以便rec.pnamepersontable[rec.personid].pname; 但是具有智能字段的namedtuples也将具有其他用途。)

""" 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

2
你难道没有在自己的问题中找到答案吗? - Torsten Marek
我不理解这个问题。也许你想让属性显示在元组中?如果你想要这样,可以重写getitem方法。 - Pepijn
1
我也感到困惑。你似乎正在做你应该做的事情,以获得你所询问的效果。你遇到了什么问题? - Omnifarious
抱歉,表述不够清晰:寻找一个生成器或属性工厂来处理半打表格,其中一些表格具有20个字段。 - denis
既然您已经更新了问题,那就是一个有趣的问题,我会稍微思考一下。 - Omnifarious
3个回答

19

你的问题的答案是:你正在做的方式就是正确的!你遇到了什么错误?为了看到一个更简单的例子,

如何使用额外的@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)

-2

这里有一种方法,使用一些语言: 将其转换为类似上面的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

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接