詳解Python with/as使用說明
with/as
使用open打開過文件的對with/as都已經(jīng)非常熟悉,其實(shí)with/as是對try/finally的一種替代方案。
當(dāng)某個對象支持一種稱為"環(huán)境管理協(xié)議"的協(xié)議時(shí),就會通過環(huán)境管理器來自動執(zhí)行某些善后清理工作,就像finally一樣:不管中途是否發(fā)生異常,最終都會執(zhí)行某些清理操作。
用法:
with expression [as var]: with_block_code
當(dāng)expression返回的對象是支持環(huán)境管理協(xié)議的時(shí)候,就可以使用with。as var是可選的,如果不使用as var,expression返回對象將被丟棄,如果使用as var,就會將expression的返回對象賦值給變量var。
整個流程大致如下:先評估expression,如果支持環(huán)境管理協(xié)議,然后開始with/as語句塊結(jié)構(gòu),當(dāng)準(zhǔn)備退出with語句塊的時(shí)候,將執(zhí)行對象中定義的善后操作。工作機(jī)制的細(xì)節(jié)見下文。
例如,open()返回的文件對象是支持環(huán)境管理協(xié)議的,所以可以用with/as來安全地打開文件:
with open(r'd:\a\b\c\a.log') as logfile: for line in logfile: print(line) ...more code here...
整個過程是先open(),然后with/as,輸出每一行后將要退出with語句塊的時(shí)候,環(huán)境管理器根據(jù)文件對象中定義的操作關(guān)閉文件。
它實(shí)際上等價(jià)于:
myfile = open(r'd:\a\b\c\a.log') try: for line in myfile: print(line) ...more code here... finally: myfile.close()
雖然在文件不被引用之后,垃圾回收器會自動回收這個文件對象,但是垃圾回收器的回收操作是有等待時(shí)間的。換句話說,如果不使用with/as打開文件,也不顯示close()關(guān)閉文件,那么這個文件很可能會在用完之后保持空閑一段時(shí)間,然后才被垃圾回收器回收。
with/as不僅用于文件打開/關(guān)閉,鎖操作也支持環(huán)境管理協(xié)議,也就是說,在有需要的時(shí)候會自動釋放鎖資源。
嵌套多個環(huán)境管理器
在python 3.1之后,with as支持多個環(huán)境管理器,使用逗號隔開即可。
with A() as a, B() as b: ...statements...
它等價(jià)于嵌套的with:
with A() as a: with B() as b: ...statements...
多環(huán)境管理器管理的多個對象會在with語句塊中出現(xiàn)異常的時(shí)候,或者執(zhí)行完with語句塊的時(shí)候全部自動被清理(例如文件關(guān)閉操作)。
例如,打開兩個文件,將它們的內(nèi)容通過zip()合并在一起,并且同時(shí)關(guān)閉它們:
with open('a.file') as f1, open('b.file') as f2: for pair in zi[(f1, f2): print(pair)
自定義環(huán)境管理器
無論是文件還是鎖,都是別人已經(jīng)寫好了環(huán)境管理器的對象。我們自己也可以寫環(huán)境管理器,讓它可以使用with/as,這實(shí)際上屬于運(yùn)算符重載的范疇。
要寫自己的環(huán)境管理器,先了解with/as的工作機(jī)制的細(xì)節(jié):
- 先評估expression,評估的返回結(jié)果是一個對象,這個對象要具有
__enter__
和__exit__
方法,返回的對象稱為"環(huán)境管理器" - 然后調(diào)用環(huán)境管理器的
__enter__
方法。__enter__
方法的返回值賦值給 as 指定的變量,或者直接丟棄(沒有使用as) - 然后執(zhí)行with語句塊中的內(nèi)容
- 如果執(zhí)行with語句塊中的內(nèi)容時(shí)拋出了異常,將調(diào)用
__exit__(type,value,traceback)
方法,其中這3個和異常相關(guān)的參數(shù)來源于sys.exc_info
。如果__exit__
返回值為False,則會自動重新拋異常以便傳播異常,否則異常被認(rèn)為合理處理 - 如果with語句塊中的內(nèi)容沒有拋異常,則直接調(diào)用
__exit__(None,None,None)
,即這三個參數(shù)都傳遞為None值
看一個簡單的示例:
class TraceBlock: def message(self, arg): print('running ' + arg) def __enter__(self): print('starting with block') return self def __exit__(self, exc_type, exc_value, exc_tb): if exc_type is None: print('exited normally\n') else: print('raise an exception! ' + str(exc_type)) return False
上面的 __enter__
方法返回的對象會賦值給as關(guān)鍵字指定的變量,在這個示例中即將對象自身返回。如果有需求,可以返回其它對象。
上面的 __exit__
中,如果異常的類型為None,說明with語句塊中的語句執(zhí)行過程沒有拋異常,正常結(jié)束即可。但是如果有異常,則要求返回False,實(shí)際上上面的 return False
可以去掉,因?yàn)楹瘮?shù)沒有return時(shí)默認(rèn)返回None,它的布爾值代表的就時(shí)False。
測試下:
with TraceBlock() as action: action.message("test 1") print("reached") print('-' * 20, "\n") with TraceBlock() as action: action.message("test 2") raise TypeError print("not reached")
結(jié)果如下:
starting with block
running test 1
reached
exited normally--------------------
starting with block
running test 2
raise an exception! <class 'TypeError'>
Traceback (most recent call last):
File "g:/pycode/list.py", line 23, in <module>
raise TypeError
TypeError
定義環(huán)境管理器不是件簡單的事。一般來說,如果不是很復(fù)雜的需求,直接使用try/finally來定義相關(guān)操作即可。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
python?numpy?中l(wèi)inspace函數(shù)示例詳解
這篇文章主要介紹了python?numpy?中l(wèi)inspace函數(shù),本文我們通過示例學(xué)習(xí)了linspace函數(shù),如果你熟悉NumPy,一定也注意到還有np.arange函數(shù),兩者最大差異是,linspace能夠精確控制終止值終值,而arange能夠更直接地控制序列中值之間的增量,需要的朋友可以參考下2023-03-03Python對圖片進(jìn)行resize、裁剪、旋轉(zhuǎn)、翻轉(zhuǎn)問題
這篇文章主要介紹了Python對圖片進(jìn)行resize、裁剪、旋轉(zhuǎn)、翻轉(zhuǎn)問題,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-05-05Python入門教程3. 列表基本操作【定義、運(yùn)算、常用函數(shù)】
這篇文章主要介紹了Python列表基本操作,結(jié)合實(shí)例形式總結(jié)分析了Python針對列表的基本定義、判斷、運(yùn)算及各種常用函數(shù)與相關(guān)使用技巧,需要的朋友可以參考下2018-10-10python清除指定目錄內(nèi)所有文件中script的方法
這篇文章主要介紹了python清除指定目錄內(nèi)所有文件中script的方法,涉及Python針對文件、字符串及正則匹配操作的相關(guān)技巧,需要的朋友可以參考下2015-06-06Python Pandas分組聚合的實(shí)現(xiàn)方法
這篇文章主要介紹了Python Pandas分組聚合的實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-07-07python lambda表達(dá)式(匿名函數(shù))寫法解析
這篇文章主要介紹了python lambda表達(dá)式(匿名函數(shù))寫法解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-09-09