描述符就是实现了描述符协议的对象,描述符协议包含三个方法:get、set__和__delete。
只实现了__get__方法的对象称为非数据描述符,这类描述符只能读取对象属性;
同时实现了__get__和__set__方法的对象是数据描述符,
接下来定义一个描述符对象:
class Example(object): """ 描述符 """ def __init__(self,name="django"): self.name=name def __get__(self,instance,owner): return self.name def __set__(self,instance,value): self.name=value class A(object): x=Example() a="a" class B(object): def __init__(self): self.x=Example() b=B() print(A.x+"\n",b.x)
可以看到,访问A.x直接输出了描述符对象的name属性。这是因为描述符作为属性访问是被自动调用的,且对于类属性和类实例属性,有着不同的调用规则.
(1)描述符对象作为类属性:Class.x将被转换为:
Class.__dict__["x"].__get__(None,Class)
2)描述符对象作为实例属性:object.x将被转换为:
type(object).__dict__["x"].__get__(object,type(object))
从输出可以看到,访问实例属性并没有调用__get__方法,而是直接返回了这个描述符实例对象。这是因为根据调用规则,type(b).dict[‘x’]是不存在的(会抛出KeyError错误),所以,不会访问后面的__get__方法。
简单地说,描述符的主要作用就是对属性的操作过程(获取、设置和删除)进行拦截,给用户自己定义操作属性行为的机会。也就是说,如果用户想“控制”属性,那么就可以考虑使用描述符。
看下面例子,通过描述符类,限制一个类的属性x只能被赋值为整数.
class Integer(object): def __ini__(self): self.value=0 def __get__(self,instance,owner): return self.value def __set__(self,instance,value): if not isinstance(value,int): raise TypeError("value must be int") self.value=value class C(object): x=Integer() c=C() # c.x="10" c.x=10
Django中规定,只有Model对象可以使用objects(查询管理器),Model对象实例是不允许的,这里就是借助描述符的特性做的实现。