我想以这种方式定义一个类:
class List(Base):
hp = Column(int,...)
name = Column(str,...)
此类表示一个列表,我可以定义/修改/编码Base
和Column
类。
有一种方法可以知道我定义属性hp / names的顺序吗?
例如,我想定义一个可以执行此操作的方法:
l = List()
l.show()
[(hp,16),(name,roger)] # in this order
答案 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
的来源
答案 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__
不起作用,因为我认为它们只适用于实例,而不适用于类属性。