Python 中的上下文管理器
with
expression
[
as
target
]
:
with
-
body
上下文管理器是為with 語句而生。只要實現(xiàn)了上下文管理器協(xié)議
__enter__
與
__exit__
,就可以使用with語句。
__enter__
通常執(zhí)行一些初始化操作,并且該函數(shù)的返回值會賦值給可選的
as target
中的
target
變量。
__exit__
執(zhí)行資源清理工作。它接收三個參數(shù),異常類型,異常實例,和異常棧,根據(jù)這些異常信息,
__exit__
可以選擇進行相應(yīng)的異常處理,并默認拋出異常。如果我們在讓
__exit__
返回True,相當于告訴python:這些異常我都已經(jīng)處理了,都在掌控之中,您老不必操心。
除了自定義類手動實現(xiàn)兩個特殊方法外,還有另一種途徑實現(xiàn)一個上下文管理器。
標準庫
contextlib
中提供了一個
@contextmanager
可以方便的把一個協(xié)程函數(shù)包裝成一個上下文管理器。
《Fluent Python》 書中一個好玩的例子:
@contextmanager
def
f
(
)
:
import
sys
print
(
'歡迎來到鏡像的世界'
)
origin_print
=
sys
.
stdout
.
write
sys
.
stdout
.
write
=
lambda
x
:
origin_print
(
x
[
:
:
-
1
]
)
# 初始化:替換系統(tǒng)輸入。運行中動態(tài)修改、添加類的方法————猴子補丁。
yield
'這里的打印都是反向輸出'
sys
.
stdout
.
write
=
origin_print
# 退出時:恢復(fù)系統(tǒng)輸入
print
(
'Finally I come back'
)
mirror_world
=
f
(
)
with
mirror_world
as
target
:
print
(
target
)
輸出結(jié)果:
歡迎來到鏡像的世界
出輸向反是都印打的里這
Finally I come back
協(xié)程函數(shù)中yield之前的所有代碼相當于
__enter__
部分的工作,執(zhí)行初始化,執(zhí)行中動態(tài)替換了系統(tǒng)的輸出功能(猴子補丁特性)。
并且把一個結(jié)果綁定到
with...as target
的
target
。至此協(xié)程函數(shù)交出代碼執(zhí)行權(quán),python轉(zhuǎn)而去執(zhí)行with-block里面的代碼。執(zhí)行完with-block 開始執(zhí)行yield之后的代碼——相當于
__exit__
的工作,執(zhí)行資源清理。
至此我們好像實現(xiàn)了一個功能正常的上下文管理器。但別忘了還有異常捕獲的機制。。。
在終端中執(zhí)行
mirror_world
時,如果with-block中拋出了一個異常,會導(dǎo)致資源清理工作沒有進行,之后所有的print仍是反向輸出。我們還應(yīng)做的是把yield行的代碼包裹在一個
try...except...finally
中,在finally-bolck中執(zhí)行資源清理工作,以保證正常退出(鬼知道用戶會在with-block搞什么蛇皮…)。
更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主
微信掃碼或搜索:z360901061
微信掃一掃加我為好友
QQ號聯(lián)系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元

