Python 类定义中的属性赋值顺序

Python 类定义中的属性赋值顺序,python,Python,我想这样定义一个类: class List(Base): hp = Column(int,...) name = Column(str,...) 这个类表示一个列表,我可以定义/修改/编码Base和列类。 有一种方法可以知道我定义属性hp/names的顺序吗? 例如,我想定义一个可以实现此目的的方法: l = List() l.show() [(hp,16),(name,roger)] # in this order 我想你想要这样的东西: import inspect c

我想这样定义一个类:

class List(Base):
    hp = Column(int,...)
    name = Column(str,...)
这个类表示一个列表,我可以定义/修改/编码
Base
类。 有一种方法可以知道我定义属性hp/names的顺序吗? 例如,我想定义一个可以实现此目的的方法:

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

我想你想要这样的东西:

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)]
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

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

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

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)]
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的源代码,类和实例属性本质上是无序的,因为它们存储在字典中

不过有几个选择:

1) 用定义的类不使用字典,属性的空间是预先分配的。我假设属性因此是有序的,因此原则上应该可以获取数据。不过,使用插槽存在一些问题,因此不推荐使用

2) 使用属性顺序向类添加列表/元组属性

3) 使用collection.namedtuple对象存储属性。namedtuple是一个元组,您可以在其中为元素命名,因此类似于C结构。这只适用于Python2.6以后的版本。由于元组是不可变的,一旦创建,您就无法修改属性-这可能是问题,也可能不是问题,取决于类

编辑:


反思
\uuuu插槽\uuuuu
将不起作用,因为我认为它们只适用于实例,而不适用于类属性。

非常感谢,这就是我要找的!哎呀,我忘了对元素进行排序,这就是重点!我修改了答案。