1. 程式人生 > >python 屬性描述符及屬性查找順序

python 屬性描述符及屬性查找順序

ron pos name esc str ret 方法 基類 類定義

 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

python 屬性描述符及屬性查找順序