內建屬性
"teachclass.py"
class Person(object):
pass
python3.5中類的內建屬性和方法
經典類(舊式類),早期如果沒有要繼承的父類,繼承裡空著不寫的類
#py2中無繼承父類,稱之經典類,py3中已默認繼承object
class Person:
pass
子類沒有實現__init__方法時,默認自動調用父類的。 如定義__init__方法時,需自己手動調用父類的 __init__方法
常用專有屬性說明觸發方式
__init__構造初始化函數創建實例後,賦值時使用,在__new__後
__new__生成實例所需屬性創建實例時
__class__實例所在的類實例.__class__
__str__實例字符串表示,可讀性print(類實例),如沒實現,使用repr結果
__repr__實例字符串表示,準確性類實例 回車 或者 print(repr(類實例))
__del__析構del刪除實例
__dict__實例自定義屬性vars(實例.__dict__)
__doc__類文檔,子類不繼承help(類或實例)
__getattribute__屬性訪問攔截器訪問實例屬性時
__bases__類的所有父類構成元素類名.__bases__
__getattribute__例子:
class Itcast(object):
def __init__(self,subject1):
self.subject1 = subject1
self.subject2 = 'cpp'
#屬性訪問時攔截器,打log
def __getattribute__(self,obj):
if obj == 'subject1':
print('log subject1')
return 'redirect python'
else: #測試時注釋掉這2行,將找不到subject2
return object.__getattribute__(self,obj)
def show(self):
print('this is Itcast')
s = Itcast("python")
print(s.subject1)
print(s.subject2)
運行結果:
log subject1
redirect python
cpp
__getattribute__的坑
class Person(object):
def __getattribute__(self,obj):
print("---test---")
if obj.startswith("a"):
return "hahha"
else:
return self.test
def test(self):
print("heihei")
t.Person()
t.a #返回hahha
t.b #會讓程序死掉
#原因是:當t.b執行時,會調用Person類中定義的__getattribute__方法,但是在這個方法的執行過程中
#if條件不滿足,所以 程序執行else裡面的代碼,即return self.test 問題就在這,因為return 需要把
#self.test的值返回,那麼首先要獲取self.test的值,因為self此時就是t這個對象,所以self.test就是
#t.test 此時要獲取t這個對象的test屬性,那麼就會跳轉到__getattribute__方法去執行,即此時產
#生了遞歸調用,由於這個遞歸過程中 沒有判斷什麼時候推出,所以這個程序會永無休止的運行下去,又因為
#每次調用函數,就需要保存一些數據,那麼隨著調用的次數越來越多,最終內存吃光,所以程序 崩潰
#
# 注意:以後不要在__getattribute__方法中調用self.xxxx