9.14 捕获类的属性定义顺序

问题

你想自动记录一个类中属性和方法定义的顺序,然后可以利用它来做很多操作(比如序列化、映射到数据库等等)。

解决方案

利用元类可以很容易的捕获类的定义信息。下面是一个例子,使用了一个OrderedDict来记录描述器的定义顺序:

  1. from collections import OrderedDict
  2.  
  3. # A set of descriptors for various types
  4. class Typed:
  5. _expected_type = type(None)
  6. def __init__(self, name=None):
  7. self._name = name
  8.  
  9. def __set__(self, instance, value):
  10. if not isinstance(value, self._expected_type):
  11. raise TypeError('Expected ' + str(self._expected_type))
  12. instance.__dict__[self._name] = value
  13.  
  14. class Integer(Typed):
  15. _expected_type = int
  16.  
  17. class Float(Typed):
  18. _expected_type = float
  19.  
  20. class String(Typed):
  21. _expected_type = str
  22.  
  23. # Metaclass that uses an OrderedDict for class body
  24. class OrderedMeta(type):
  25. def __new__(cls, clsname, bases, clsdict):
  26. d = dict(clsdict)
  27. order = []
  28. for name, value in clsdict.items():
  29. if isinstance(value, Typed):
  30. value._name = name
  31. order.append(name)
  32. d['_order'] = order
  33. return type.__new__(cls, clsname, bases, d)
  34.  
  35. @classmethod
  36. def __prepare__(cls, clsname, bases):
  37. return OrderedDict()

在这个元类中,执行类主体时描述器的定义顺序会被一个 OrderedDict捕获到, 生成的有序名称从字典中提取出来并放入类属性_order 中。这样的话类中的方法可以通过多种方式来使用它。例如,下面是一个简单的类,使用这个排序字典来实现将一个类实例的数据序列化为一行CSV数据:

  1. class Structure(metaclass=OrderedMeta):
  2. def as_csv(self):
  3. return ','.join(str(getattr(self,name)) for name in self._order)
  4.  
  5. # Example use
  6. class Stock(Structure):
  7. name = String()
  8. shares = Integer()
  9. price = Float()
  10.  
  11. def __init__(self, name, shares, price):
  12. self.name = name
  13. self.shares = shares
  14. self.price = price

我们在交互式环境中测试一下这个Stock类:

  1. >>> s = Stock('GOOG',100,490.1)
  2. >>> s.name
  3. 'GOOG'
  4. >>> s.as_csv()
  5. 'GOOG,100,490.1'
  6. >>> t = Stock('AAPL','a lot', 610.23)
  7. Traceback (most recent call last):
  8. File "<stdin>", line 1, in <module>
  9. File "dupmethod.py", line 34, in __init__
  10. TypeError: shares expects <class 'int'>
  11. >>>

讨论

本节一个关键点就是OrderedMeta元类中定义的 __prepare__() 方法。这个方法会在开始定义类和它的父类的时候被执行。它必须返回一个映射对象以便在类定义体中被使用到。我们这里通过返回了一个OrderedDict而不是一个普通的字典,可以很容易的捕获定义的顺序。

如果你想构造自己的类字典对象,可以很容易的扩展这个功能。比如,下面的这个修改方案可以防止重复的定义:

  1. from collections import OrderedDict
  2.  
  3. class NoDupOrderedDict(OrderedDict):
  4. def __init__(self, clsname):
  5. self.clsname = clsname
  6. super().__init__()
  7. def __setitem__(self, name, value):
  8. if name in self:
  9. raise TypeError('{} already defined in {}'.format(name, self.clsname))
  10. super().__setitem__(name, value)
  11.  
  12. class OrderedMeta(type):
  13. def __new__(cls, clsname, bases, clsdict):
  14. d = dict(clsdict)
  15. d['_order'] = [name for name in clsdict if name[0] != '_']
  16. return type.__new__(cls, clsname, bases, d)
  17.  
  18. @classmethod
  19. def __prepare__(cls, clsname, bases):
  20. return NoDupOrderedDict(clsname)

下面我们测试重复的定义会出现什么情况:

  1. >>> class A(metaclass=OrderedMeta):
  2. ... def spam(self):
  3. ... pass
  4. ... def spam(self):
  5. ... pass
  6. ...
  7. Traceback (most recent call last):
  8. File "<stdin>", line 1, in <module>
  9. File "<stdin>", line 4, in A
  10. File "dupmethod2.py", line 25, in __setitem__
  11. (name, self.clsname))
  12. TypeError: spam already defined in A
  13. >>>

最后还有一点很重要,就是在 new() 方法中对于元类中被修改字典的处理。尽管类使用了另外一个字典来定义,在构造最终的 class 对象的时候,我们仍然需要将这个字典转换为一个正确的 dict 实例。通过语句 d = dict(clsdict) 来完成这个效果。

对于很多应用程序而已,能够捕获类定义的顺序是一个看似不起眼却又非常重要的特性。例如,在对象关系映射中,我们通常会看到下面这种方式定义的类:

  1. class Stock(Model):
  2. name = String()
  3. shares = Integer()
  4. price = Float()

在框架底层,我们必须捕获定义的顺序来将对象映射到元组或数据库表中的行(就类似于上面例子中的 as_csv() 的功能)。这节演示的技术非常简单,并且通常会比其他类似方法(通常都要在描述器类中维护一个隐藏的计数器)要简单的多。

原文:

http://python3-cookbook.readthedocs.io/zh_CN/latest/c09/p14_capture_class_attribute_definition_order.html