Python上下文管理器Content Manager
在 Python 中,我們會經(jīng)常聽到上下文管理器(Context Manager),那我們探討下這是什么,又有什么功能。
在 Python 中的上下文管理器中,使用 with 打開文件是使用最多的,其中離開 with 包含的語句后會執(zhí)行一些類似于清理的工作,如關(guān)閉文件,關(guān)閉連接對象等操作。
實(shí)踐
我們在代碼實(shí)踐的時候,忽略了在同一代碼片段中,先打開文件,然后直接對文件進(jìn)行其他處理,因?yàn)檫@樣沒有任何意義,資源是處于被占用的情況。
先看下面檢測的代碼:
#!/usr/bin/env python # _*_ coding: UTF-8 _*_ # MedusaSorcerer Script import os class OpenFile: def __init__(self): self.file = None def open(self, path): self.file = open(path, 'w') if __name__ == '__main__': file_path = 'medusa.md' file = OpenFile() file.open(file_path) os.remove(file_path)
代碼中我們把文件對象,進(jìn)行了實(shí)例屬性的方式引用,在此之后,我們使用 os 模塊進(jìn)行刪除被寫入的文件。執(zhí)行改代碼片段后,會出現(xiàn)以下內(nèi)容:
Traceback (most recent call last):
File "medusa/main.py", line 19, in <module>
os.remove(file_path)
PermissionError: [WinError 32] 另一個程序正在使用此文件,進(jìn)程無法訪問。: 'medusa.md'Process finished with exit code 1
那是因?yàn)楸粍h除的文件沒有得到資源釋放。我們在上面的基礎(chǔ)上進(jìn)行套用函數(shù)的方式:
#!/usr/bin/env python # _*_ coding: UTF-8 _*_ # MedusaSorcerer Script import os class OpenFile: def __init__(self): self.file = None def open(self, path): self.file = open(path, 'w') def open_file(path): file = OpenFile() file.open(path) if __name__ == '__main__': file_path = 'medusa.md' open_file(file_path) os.remove(file_path)
這段代碼會成功的被執(zhí)行成功,原因是當(dāng)你執(zhí)行函數(shù)的時候,函數(shù)內(nèi)的臨時變量將被回收釋放,因此 OpenFile 的實(shí)例對象被釋放了,實(shí)例屬性也就不存在而被釋放,所以會執(zhí)行成功。
那是否我們的操作都應(yīng)該使用函數(shù)包裹的方式執(zhí)行呢?with 的出現(xiàn),完美解決了這個問題:
#!/usr/bin/env python # _*_ coding: UTF-8 _*_ # MedusaSorcerer Script import os if __name__ == '__main__': file_path = 'medusa.md' with open(file_path, 'w') as f: print(f) os.remove(file_path)
在 with 語法中,將后面打開文件的操作,返回的文件對象,賦值給 f 變量,在結(jié)構(gòu)體中輸出了 f 變量的內(nèi)容,并且在結(jié)構(gòu)體外刪除了該文件:
medusa\python.exe medusa/main.py
<_io.TextIOWrapper name='medusa.md' mode='w' encoding='cp936'>Process finished with exit code 0
在沒有使用 close() 的情況下,依舊可以對文件進(jìn)行刪除,這就是上下文管理的美妙。
實(shí)現(xiàn)
上下文管理,實(shí)際上是實(shí)現(xiàn)了 __enter__ 和 __exit__ 方法:
#!/usr/bin/env python # _*_ coding: UTF-8 _*_ # MedusaSorcerer Script class Medusa: def __init__(self): print('__init__') def __enter__(self): print('__enter__') def __exit__(self, exc_type, exc_val, exc_tb): print('__exit__') if __name__ == '__main__': medusa = Medusa() with medusa: print('with object') print('finish')
以下是輸出結(jié)果:
__init__
__enter__
with object
__exit__
finish
我們發(fā)現(xiàn)魔法方法在結(jié)合某些語法后會發(fā)生自動調(diào)度,所以,上下文管理中就在自動調(diào)度中,關(guān)閉了某些對象。
優(yōu)點(diǎn)
實(shí)現(xiàn)上下文管理可以簡化我們的代碼,讓代碼更加簡單易讀,使用最少的代碼量,就可以完成全部工作。
到此這篇關(guān)于Python上下文管理器Content Manager的文章就介紹到這了,更多相關(guān)Python上下文管理器內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python實(shí)現(xiàn)excel轉(zhuǎn)置問題詳解
這篇文章主要介紹了python實(shí)現(xiàn)excel轉(zhuǎn)置問題詳解,excel轉(zhuǎn)置分為兩種情況,一個是較為簡單的只需要行轉(zhuǎn)列,列轉(zhuǎn)行,具體詳解,感興趣的小伙伴可以參考一下2022-09-09python requests 庫請求帶有文件參數(shù)的接口實(shí)例
今天小編就為大家分享一篇python requests 庫請求帶有文件參數(shù)的接口實(shí)例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-01-01python實(shí)現(xiàn)簡易動態(tài)時鐘
這篇文章主要為大家詳細(xì)介紹了python實(shí)現(xiàn)簡易動態(tài)時鐘,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-11-11Pytorch加載部分預(yù)訓(xùn)練模型的參數(shù)實(shí)例
今天小編就為大家分享一篇Pytorch加載部分預(yù)訓(xùn)練模型的參數(shù)實(shí)例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-08-08Python使用Matplotlib實(shí)現(xiàn)雨點(diǎn)圖動畫效果的方法
這篇文章主要介紹了Python使用Matplotlib實(shí)現(xiàn)雨點(diǎn)圖動畫效果的方法,結(jié)合實(shí)例形式分析了win10安裝ffmpeg及animation函數(shù)的使用相關(guān)操作技巧,需要的朋友可以參考下2017-12-12