首先,我們來看一下with的語法格式:
withcontext_expression[astarget(s)]:with-body
with語法非常簡單,我們只需要with一個表達式,然后就可以執行自定義的業務邏輯。
但是,with后面的表達式是可以任意寫的嗎?
答案是否定的。要想使用with語法塊,with后面的的對象需要實現「上下文管理器協議」。
什么是「上下文管理器協議」?
一個類在Python中,只要實現以下方法,就實現了「上下文管理器協議」:
__enter__:在進入with語法塊之前調用,返回值會賦值給with的target
__exit__:在退出with語法塊時調用,一般用作異常處理
我們來看實現了這2個方法的例子:
classTestContext:def__enter__(self):print('__enter__')return1def__exit__(self,exc_type,exc_value,exc_tb):print('exc_type:%s'%exc_type)print('exc_value:%s'%exc_value)print('exc_tb:%s'%exc_tb)withTestContext()ast:print('t:%s'%t)#Output:#__enter__#t:1#exc_type:None#exc_value:None#exc_tb:None
在這個例子中,我們定義了TestContext類,它分別實現了__enter__和exit方法。
這樣一來,我們就可以把TestContext當做一個「上下文管理器」來使用,也就是通過withTestContext()ast方式來執行。
從輸出結果我們可以看到,具體的執行流程如下:
__enter__在進入with語句塊之前被調用,這個方法的返回值賦給了with后的t變量
__exit__在執行完with語句塊之后被調用
如果在with語句塊內發生了異常,那么__exit__方法可以拿到關于異常的詳細信息:
exc_type:異常類型
exc_value:異常對象
exc_tb:異常堆棧信息
我們來看一個發生異常的例子,觀察__exit__方法拿到的異常信息是怎樣的:
withTestContext()ast:#這里會發生異常a=1/0print('t:%s'%t)#Output:#__enter__#exc_type:#exc_value:integerdivisionormodulobyzero#exc_tb:#Traceback(mostrecentcalllast):#File"base.py",line16,in#a=1/0#ZeroDivisionError:integerdivisionormodulobyzero
從輸出結果我們可以看到,當with語法塊內發生異常后,__exit__輸出了這個異常的詳細信息,其中包括異常類型、異常對象、異常堆棧。
如果我們需要對異常做特殊處理,就可以在這個方法中實現自定義邏輯。
回到最開始我們講的,使用with讀取文件的例子。之所以with能夠自動關閉文件資源,就是因為內置的文件對象實現了「上下文管理器協議」,這個文件對象的__enter__方法返回了文件句柄,并且在__exit__中實現了文件資源的關閉,另外,當with語法塊內有異常發生時,會拋出異常給調用者。
偽代碼可以這么寫:
classFile:def__enter__(self):returnfile_objdef__exit__(self,exc_type,exc_value,exc_tb):#with退出時釋放文件資源file_obj.close()#如果with內有異常發生拋出異常ifexc_typeisnotNone:raiseexception
這里我們小結一下,通過對with的學習,我們了解到,with非常適合用需要對于上下文處理的場景,例如操作文件、Socket,這些場景都需要在執行完業務邏輯后,釋放資源。
以上內容為大家介紹了python的上下文管理器,希望對大家有所幫助,如果想要了解更多Python相關知識,請關注IT培訓機構:千鋒教育。http://www.dietsnews.net/