1 import numbers 2 class IntField: # 当一个类实现了 __get__, __set__, __delete__ 3 """ 4 数据描述符 5 """ 6 def __get__(self, instance, owner): 7 return self.value 8 def __set__(self, instance, value): 9 if not isinstance(value, numbers.Integral): 10 raise ValueError("int value need") 11 if value < 0: 12 raise ValueError("positive value need") 13 self.value = value 14 def __delete__(self, instance): 15 pass 16 17 class NonDataIntField: 18 """ 19 非数据属性描述符 20 """ 21 def __get__(self, instance, owner): 22 return self.value 23 24 class User: 25 age = IntField() 26 27 if __name__ == "__main__": 28 u = User() 29 u.age
如果 user 是某个类的实例,那么user.age(以及等价的 getattr(user, 'age') )
首先调用 __getattribute__。如果类定义了 __getattr__ 方法,那么在 __getattribute__ 抛出 AttributeError 的时候就会调用到 __getattr__,
而对于描述符 (__get__) 的调用,则是发生在 __getattribute__ 内部的。
user = User(), 那么 user.age 顺序如下:
1、如果"age"是出现在 User 类或其基类的 __dict__ 中,且 age 是 data descriptor (数据属性描述符), 那么调用其 __get__ 方法
2、如果 "age" 出现在实例 user 的 __dict__ 中,那么直接返回 obj.__dict__['age'],
3‘、如果 "age" 出现在 User 类或其基类的 __dict__ 中
3.1、如果 "age" 是 non-data descriptor (), 那么调 用其 __get__ 方法
3.2、返回 __dict__['age']
4、如果 User 有 __getattr__ 方法, 调用 __getattr__ 方法,
5、抛出 AttributeError