Python描述符

2021/11/27 17:11:28

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



这篇关于Python描述符的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!


扫一扫关注最新编程教程