python之属性描述符

做为一个小白,天天都在不断地看东西,学知识,今天给你们介绍一个好东西——属性描述符
什么是属性描述符呢?
其实在一个类中实现set__、__get__、__delete中任意一个魔法函数就是一个属性描述符。
接下来咱们定义一个属性描述符:函数

class IntegerField:
    def __get__(self, instance, owner):
        pass

    def __set__(self, instance, value):
        pass

    def __delete__(self, instance):
        pass


class User:
    high= IntField()

__get__:当咱们用类或者实例来调用该属性时,会返回__get__函数的结果。
__set__:当咱们用实例来设置属性值时,Python会调用该函数。对类没有限制做用。
__delete__:当咱们用实例试图删除该属性时,Python会调用该函数。对类没有限制做用。
__set__中参数:self->描述符实例, instance->托管实例, value->设置值
__get__中参数:self->描述符实例, instance->托管实例, owner->托管类的引用
到底这个东西怎么用呢?接下来为你们修改上面的代码code

class IntegerField:
    def __get__(self, instance, owner):
        return self.value
        
    def __set__(self, instance, value):
        if not isinstance(value,numbers.Integral):
            raise ValueError("请输入一个整数")
        self.value=value
        
    def __delete__(self, instance):
        pass


class User:
    high=IntegerField()
    
#验证代码
if __name__ == '__main__':
    user=User()
    user.high='175'    #报错,ValueError:请输入一个整数
    User.high=175    #正确执行,不报错

这样咱们就可以运用属性描述符来给属性附上必定的逻辑了。
其实在属性描述符下还分为
一、数据描述符:实现了__set__、__get__get

if __name__ == '__main__':
    user=User()
    user.high=175
    print(user.__dict__)    #high是不放入__dict__中的,优先查找数据描述符中的值
    user.__dict__["high"]="abc"    #这样赋值时能够的,而且能够放入__dict__中
    print(user.high)    #会报错,由于在调用__get__方法时并无value属性

二、非数据:实现__get__不实现__set__it

class NonField:
    def __init__(self, high=170):
        self.value = high

    def __get__(self, instance, owner):
        return self.value


class User:
    high = NonField()


if __name__ == '__main__':
    user = User()
    user.high = '175'    #会放入user.__dict__中
    print(user.__dict__)