1

作为一个小白,每天都在不断地看东西,学知识,今天给大家介绍一个好东西——属性描述符
什么是属性描述符呢?
其实在一个类中实现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->托管类的引用
到底这个东西怎么用呢?接下来为大家修改上面的代码

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    #正确执行,不报错

这样我们就能够运用属性描述符来给属性附上一定的逻辑了。
其实在属性描述符下还分为
1、数据描述符:实现了__set__、__get__

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

2、非数据:实现__get__不实现__set__

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__)    

二十一
1.4k 声望867 粉丝

无论遇到多大的困难,你总是能扛过去,坚持一件事,对自己