Python面向對象編程Day 28部分
- 2020 年 1 月 19 日
- 筆記
__enter__和__exit__
with obj as f:
'代碼塊'
1.with obj —>觸發obj.__enter__(),拿到返回值
2.as f —> f=返回值
3.with obj as f 等同於 f=obj.__enter__()
4.執行代碼塊
兩種情況:
沒有異常的情況下,整個代碼塊運行完畢後去觸發__exit__,它的三個參數都為None
有異常的情況下,會從異常出現的位置直接觸發__exit__,此時分兩種情況:
如果__exit__的返回值為True,代表吞掉了異常
如果__exit__的返回值不為True,代表吐出了異常
(exit的運行完畢就代表了整個with語句的執行完畢,異常後代碼塊內的語句不會執行)
用途:使用with語句的目的是省去手動清理的過程,另外在需要管理一些資源比如文件,網絡連接和鎖的編程環境中,可以在__exit__中定製自動釋放資源的機制。
異常構成簡單了解
異常類,異常值,追蹤信息 分別對應exc_type/exc_val/exc_tb
描述符應用
1 class Typed: 2 def __init__(self,key,expected_type): 3 self.key=key 4 self.expected_type=expected_type 5 def __get__(self, instance, owner): #instance是p1實例 6 print('get方法') 7 return instance.__dict__[self.key] 8 def __set__(self, instance, value): 9 print('set方法') 10 if not isinstance(value,self.expected_type): 11 raise TypeError('%s 傳入的類型不是%s' %(self.key,self.expected_type)) 12 instance.__dict__[self.key]=value 13 def __delete__(self, instance): 14 print('delete方法') 15 instance.__dict__.pop(self.key) 16 17 class People: 18 name=Typed('name',str) #t1.__set__() self.__set__() 19 age=Typed('age',int) #t1.__set__() self.__set__() 20 def __init__(self,name,age,salary): 21 self.name=name 22 self.age=age 23 self.salary=salary 24 25 p1=People(213,13,13.3)
輸出
Traceback (most recent call last): File "G:/BaiduNetdiskDownload/第04階段-Python3面向對象編程(24-28)/python全棧s3 day028/day28課上代碼/描述符的應用.py", line 41, in <module> p1=People(213,13,13.3) File "G:/BaiduNetdiskDownload/第04階段-Python3面向對象編程(24-28)/python全棧s3 day028/day28課上代碼/描述符的應用.py", line 36, in __init__ self.name=name File "G:/BaiduNetdiskDownload/第04階段-Python3面向對象編程(24-28)/python全棧s3 day028/day28課上代碼/描述符的應用.py", line 25, in __set__ raise TypeError('%s 傳入的類型不是%s' %(self.key,self.expected_type)) TypeError: name 傳入的類型不是<class 'str'>
類的裝飾器
1 def deco(obj): 2 print('==========',obj) 3 obj.x=1 4 obj.y=2 5 obj.z=3 6 return obj 7 8 @deco #Foo=deco(Foo) 9 class Foo: 10 pass 11 12 print(Foo.__dict__)
輸出
========== <class '__main__.Foo'> {'__module__': '__main__', '__dict__': <attribute '__dict__' of 'Foo' objects>, '__weakref__': <attribute '__weakref__' of 'Foo' objects>, '__doc__': None, 'x': 1, 'y': 2, 'z': 3}
自定製property實現延遲計算功能
1 class Lazyproperty: 2 def __init__(self,func): 3 # print('==========>',func) 4 self.func=func 5 def __get__(self, instance, owner): 6 print('get') 7 if instance is None: 8 return self 9 res=self.func(instance) #函數運行的時候要把實例本身傳進去,而不是類 10 setattr(instance,self.func.__name__,res) 11 return res 12 # def __set__(self, instance, value): 13 # pass 14 15 class Room: 16 def __init__(self,name,width,length): 17 self.name=name 18 self.width=width 19 self.length=length 20 @Lazyproperty # area=Lazypropery(area) 21 def area(self): 22 return self.width * self.length 23 @property # area1=property(area1) 24 def area1(self): 25 return self.width * self.length 26 27 r1=Room('廁所',1,1) 28 print(r1.__dict__) 29 30 # 實例調用 31 print(r1.area) 32 print(r1.__dict__) 33 print(Room.__dict__) 34 35 # 類調用 被描述符代理的屬性 instance傳的是None owner不變 36 print(Room.area) 37 38 # 不再調用get方法,因為此時有實例屬性, 39 # 其優先級高於非數據描述符 若此時加上set屬性就無法實現了 40 print(r1.area) 41 print(r1.area)
輸出
{'name': '廁所', 'width': 1, 'length': 1} get 1 {'name': '廁所', 'width': 1, 'length': 1, 'area': 1} {'__module__': '__main__', '__init__': <function Room.__init__ at 0x000001EFB23EA620>, 'area': <__main__.Lazyproperty object at 0x000001EFB07D8908>, 'area1': <property object at 0x000001EFB0799688>, '__dict__': <attribute '__dict__' of 'Room' objects>, '__weakref__': <attribute '__weakref__' of 'Room' objects>, '__doc__': None} get <__main__.Lazyproperty object at 0x000001EFB07D8908> 1 1
property補充
1 class Foo: 2 @property 3 def AAA(self): 4 print('get的時候運行我啊') 5 6 # 下面兩個函數依附於靜態屬性存在 7 @AAA.setter 8 def AAA(self,val): 9 print('set的時候運行我啊',val) 10 @AAA.deleter 11 def AAA(self): 12 print('del的時候運行我啊') 13 #只有在屬性AAA定義property後才能定義AAA.setter,AAA.deleter 14 f1=Foo() 15 f1.AAA 16 f1.AAA='aaa' 17 del f1.AAA #另一種寫法,效果一樣
class Foo: def get_AAA(self): print('get的時候運行我啊') def set_AAA(self,val): print('set的時候運行我啊',val) def del_AAA(self): print('del的時候運行我啊') AAA=property(get_AAA,set_AAA,del_AAA) #順序不要變 f1=Foo() f1.AAA f1.AAA='aaa' del f1.AAA
輸出
get的時候運行我啊 set的時候運行我啊 aaa del的時候運行我啊
元類
type元類是類的類,是類的模板。元類是用來控制如何創建類的,正如類是創建對象的模板一樣。元類的實例為類,正如類的實例為對象。
1 def say_hi(name): 2 return('hello,%s'%name) 3 FFo=type('FFo',(object,),{'gender':'female','say_hi':say_hi}) 4 print(FFo) 5 print(FFo.__dict__) 6 print(FFo.say_hi('chenyuan')) 7 print('Your gender is %s'%FFo.gender)
輸出
<class '__main__.FFo'> {'gender': 'female', 'say_hi': <function say_hi at 0x000001CF9911C1E0>, '__module__': '__main__', '__dict__': <attribute '__dict__' of 'FFo' objects>, '__weakref__': <attribute '__weakref__' of 'FFo' objects>, '__doc__': None} hello,chenyuan Your gender is female
實例調用函數,會自動把實例本身傳進去當參數,而類調用函數時如果需要得寫self
自定製元類
1 class MyType(type): 2 def __init__(self,a,b,c): 3 print('元類的構造函數執行') 4 # print(a) 5 # print(b) 6 # print(c) 7 def __call__(self, *args, **kwargs): 8 # print('=-======>') 9 # print(self) 10 # print(args,kwargs) 11 obj=object.__new__(self) #object.__new__(Foo)-->f1 12 self.__init__(obj,*args,**kwargs) #Foo.__init__(f1,*arg,**kwargs) 13 return obj 14 class Foo(metaclass=MyType): #Foo=MyType(Foo,'Foo',(),{})---》__init__ 15 def __init__(self,name): 16 self.name=name #f1.name=name 17 # print(Foo) 18 # f1=Foo('alex') 19 # print(f1) 20 21 f1=Foo('alex') 22 print(f1) 23 print(f1.__dict__)
輸出
元類的構造函數執行 <__main__.Foo object at 0x0000025A13005048> {'name': 'alex'}