在類 里面,其實並沒有方法這個東西,所有的東西都保存在屬性里面,所謂的調用方法其實是類里面的一個同名屬性指向了一個函數(方法),返回的是函數的引用,再用 函數() 這種方式就可以調用它
在調用實例的方法的時候,實際上給對象傳過去的只是一個字符串而已,
比如 t = Test() , 假設 func 是 Test類的一個方法,那么 t.func() 是先給類的 __getattribute__ 方法傳過去了一個 'func' 字符串 ,經過處理之后返回的是指向 func 函數的引用 ,最后 func() 就調用了這個函數(方法)
class Test: def __init__(self): self.a = 100 self.b = 200 def __getattribute__(self, item): if item == 'a': print('------%s------' % item) return '%s is get' % item else: return object.__getattribute__(self, item) t = Test() print(t.a) >>>------a------ a is get
無論調用屬性還是方法,都是先強制調用 __getattribute__ 方法,然后再傳回屬性的值或者是 函數(方法)的引用.
注意,在 __getattribute__ 方法里面 如果再用類似 self.*** 這種調用方式 ,那么是一件很不好的行為,因為 每次調用類的屬性都會強制調用 __getattribute__ ,所以極有可能造成遞歸調用.如:
class Test: def __getattribute__(self, item): if item == 'a': print('------%s------' % item) return '%s is get' % item else: return self.test # <---------------------------------------------------------------------------- def test(self): print('.........') t = Test() print(t.b) >>>RecursionError: maximum recursion depth exceeded in comparison
self.test 會再次調用 __getattribute__ , 最后結果 就是 遞歸異常:最大遞歸深度超過什么什么....
在類的magic方法里面還有一個 __getattr__ 方法,這個方法是歸屬於 _getattribute__ 調用的,
就是 訪問屬性的時候首先會調用 __getattribute__ 方法,這個方法會檢測 __dict__ 里面有沒有這個屬性,沒有的話最后再調用一下 __getattr__
總結: 首先必須調用 __getattribute__ , 里面實現了 查找 __dict__ 里面有沒有這個鍵,如果沒有再調用 __getattr__方法.