类定义中的属性赋值顺序

时间:2010-01-06 13:09:10

标签: python

我想以这种方式定义一个类:

class List(Base):
    hp = Column(int,...)
    name = Column(str,...)

此类表示一个列表,我可以定义/修改/编码BaseColumn类。 有一种方法可以知道我定义属性hp / names的顺序吗? 例如,我想定义一个可以执行此操作的方法:

l = List()
l.show()
[(hp,16),(name,roger)] # in this order

3 个答案:

答案 0 :(得分:7)

在内部,属性定义存储在字典中,字典不保留元素的顺序。您可以更改Base类中的属性处理,或者存储创建顺序,如下所示:

class Column:
    creation_counter = 0

    def __init__(self):
            self.creation_counter = Column.creation_counter
            Column.creation_counter += 1

class Base:
    def show(self):
        fields = [(name, (obj, obj.value)) for name, obj in self.__class__.__dict__.items() if isinstance(obj, Column)] 
        fields.sort(lambda (name1, (obj1, value1)), (name2, (obj2, value2)): cmp(obj1.creation_counter, obj2.creation_counter))
        return fields

Django以类似的方式存储表单字段的顺序,尽管方式更复杂。如果您有兴趣,请查看Django formsfields

的来源

答案 1 :(得分:0)

我想你想要这样的东西:

import inspect

class List(Base):
    hp = Column(int,...)
    name = Column(str,...)

    def show(self):
        public_bits = [(x, getattr(self, x)) for x in dir(self) if not x.startswith('__')]
        return [(x, y) for (x, y) in public_bits if not inspect.ismethod(y)]

可悲的是,dir按字母顺序返回属性,因此这可能无法完全解决您的问题。

答案 2 :(得分:0)

类和实例属性本质上是无序的,因为它们存储在字典中。

但有几个选择:

1)使用__slots__定义的类不使用字典,并且预先分配属性的空间。我认为属性是有序的,因此原则上应该可以获取数据。虽然使用插槽存在问题,但不建议这样做。

2)使用属性顺序向列中添加list / tuple属性。

3)使用collection.namedtuple对象来存储属性。 namedtuple是一个元组,您可以在其中为元素指定名称,因此类似于C结构。这只适用于Python 2.6以上版本。由于元组是不可变的,因此您无法在创建后修改属性 - 这可能会也可能不会成为问题,具体取决于类。

修改

反射__slots__不起作用,因为我认为它们只适用于实例,而不适用于类属性。