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