Python描述符

  • Post author:
  • Post category:python



描述符

就是实现了描述符协议的对象,描述符协议包含三个方法:

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对象实例是不允许的,这里就是借助描述符的特性做的实现。



版权声明:本文为ma416539432原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。