python 屬性描述符及屬性查找順序
阿新 • • 發佈:2019-01-22
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 raiseValueError("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 屬性描述符及屬性查找順序