描述符就是实现了描述符协议的对象,描述符协议包含三个方法: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对象实例是不允许的,这里就是借助描述符的特性做的实现。