8.9 创建新的类或实例属性
问题
你想创建一个新的拥有一些额外功能的实例属性类型,比如类型检查。
解决方案
如果你想创建一个全新的实例属性,可以通过一个描述器类的形式来定义它的功能。
一个描述器就是一个实现了三个核心的属性访问操作(get, set, delete)的类, 分别为 __get__()
、__set__()
和 __delete__()
这三个特殊的方法。 这些方法接受一个实例作为输入,之后相应的操作实例底层的字典。
为了使用一个描述器,需将这个描述器的实例作为类属性放到一个类的定义中。例如:
当你这样做后,所有对描述器属性(比如x或y)的访问会被 __get__()
、__set__()
和 __delete__()
方法捕获到。例如:
作为输入,描述器的每一个方法会接受一个操作实例。 为了实现请求操作,会相应的操作实例底层的字典(__dict__
属性)。 描述器的 self.name
属性存储了在实例字典中被实际使用到的 key。
讨论
描述器可实现大部分Python类特性中的底层魔法, 包括 @classmethod
、@staticmethod
、@property
,甚至是 __slots__
特性。
通过定义一个描述器,你可以在底层捕获核心的实例操作(get, set, delete),并且可完全自定义它们的行为。 这是一个强大的工具,有了它你可以实现很多高级功能,并且它也是很多高级库和框架中的重要工具之一。
描述器的一个比较困惑的地方是它只能在类级别被定义,而不能为每个实例单独定义。因此,下面的代码是无法工作的:
同时,__get__()
方法实现起来比看上去要复杂得多。
__get__()
看上去有点复杂的原因归结于实例变量和类变量的不同。 如果一个描述器被当做一个类变量来访问,那么 instance
参数被设置成 None
。 这种情况下,标准做法就是简单的返回这个描述器本身即可(尽管你还可以添加其他的自定义操作)。例如:
描述器通常是那些使用到装饰器或元类的大型框架中的一个组件。同时它们的使用也被隐藏在后面。
如果你只是想简单的自定义某个类的单个属性访问的话就不用去写描述器了。 这种情况下使用8.6小节介绍的property技术会更加容易。 当程序中有很多重复代码的时候描述器就很有用了 (比如你想在你代码的很多地方使用描述器提供的功能或者将它作为一个函数库特性)。
Last updated
Was this helpful?