python 属性描述符及属性查找顺序

 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 

猜你喜欢

转载自www.cnblogs.com/sqtu/p/10301322.html