Python描述符

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值