Python with提前退出:坑與解決方案
問(wèn)題的起源
早些時(shí)候使用with實(shí)現(xiàn)了一版全局進(jìn)程鎖,希望實(shí)現(xiàn)以下效果:

全局進(jìn)程鎖本身不用多說(shuō),大部分都依靠外部的緩存來(lái)實(shí)現(xiàn)的,redis上用的是setnx,有時(shí)候根據(jù)需要加上緩存擊穿問(wèn)題、隨機(jī)延后以防止對(duì)緩存本身造成壓力。
當(dāng)時(shí)同樣寫了單元測(cè)試來(lái)測(cè)試這段代碼的有效性:

看起來(lái)非常完美地通過(guò)了。
這樣的一個(gè)全局進(jìn)程鎖是通過(guò)__enter__方法拋出異常, __exit__方法中捕獲異常來(lái)實(shí)現(xiàn)的:

看起來(lái)還不錯(cuò),畢竟單元測(cè)試都過(guò)了。
但是,這樣的實(shí)現(xiàn)是有問(wèn)題的:
原因在于__exit__ 的執(zhí)行不是包在__enter__ 之外的,因此__enter__拋出的異常,不會(huì)被__exit__捕獲。
上面的單元測(cè)試恰好通過(guò),是因?yàn)槠渲杏袃蓚€(gè)with語(yǔ)句,外面的with 捕獲的其實(shí)是里面的__enter__ 拋出的異常
使用改進(jìn)后的單元測(cè)試:

就會(huì)發(fā)現(xiàn)單元測(cè)試過(guò)不去了。
這個(gè)問(wèn)題是我試圖使用with實(shí)現(xiàn)另一個(gè)邏輯:AB測(cè)試 時(shí)出現(xiàn)的,同樣是__enter__拋出異常,__exit__ 試圖捕獲:

調(diào)試沒(méi)有通過(guò)的單元測(cè)試的時(shí)候發(fā)現(xiàn),拋出異常后根本沒(méi)有執(zhí)行到__enter__。
第一種解決方案
既然想明白了with的執(zhí)行順序,那么第一種解決方案就呼之欲出了:既然__exit__捕獲的異常在__enter__執(zhí)行完成之后,那么我們提供一個(gè)函數(shù)確認(rèn)一下就可以了,把ABContext實(shí)現(xiàn)改成這樣:

使用的時(shí)候:

但這樣的解決方法并不優(yōu)雅,萬(wàn)一使用這個(gè)ABContext的時(shí)候忘記用ensure方法了,那么就等于完全沒(méi)用這個(gè)Context方法,太容易失誤了,而且代碼也失去了Pythonic的性質(zhì)。
第二種解決方法
翻了一下contextlib的標(biāo)準(zhǔn)庫(kù)文檔,發(fā)現(xiàn)有一個(gè)已經(jīng)廢棄的函數(shù):contextlib.nested

可以執(zhí)行多個(gè)上下文:

這個(gè)廢棄的特性在Python2.7之后,可以直接由with關(guān)鍵字執(zhí)行,形如:

這個(gè)特性還不錯(cuò),根據(jù)__enter__的執(zhí)行順序的話,那么我們可以實(shí)現(xiàn)一個(gè)由第一個(gè) context的__exit__來(lái)捕獲,第二個(gè)context的__enter__來(lái)拋出異常,
如同這樣:

結(jié)合前面我們實(shí)現(xiàn)的ABContext的使用是這樣的:

good,單元測(cè)試就這樣過(guò)了!
能不能再給力點(diǎn)?
確實(shí),在with里要寫倆context有點(diǎn)蛋疼,并不是特別優(yōu)雅,能不能還是回到最初的那種用法:我們只用寫一條context,這一個(gè)context做到了兩個(gè)context的事情?
要是nested那個(gè)函數(shù)還在就好了。。要的其實(shí)就是它的功能。
Python3.1之后contextlib提供了一個(gè)ExitStack的功能來(lái)提供一個(gè)模擬的功能,但試了一下發(fā)現(xiàn),實(shí)際上只調(diào)用了__enter__方法,但沒(méi)有做對(duì)應(yīng)的異常捕獲。
第三種解決方案
哈哈哈哈把自己繞到圈子里去了,想了一下,同樣是一個(gè)縮進(jìn)的代碼塊,為什么不能用if來(lái)解決呢!不就是個(gè):

的問(wèn)題。。。
TIL
總之學(xué)到了contextlib里的一些有用的函數(shù)和裝飾器,也第一次發(fā)現(xiàn)with可以放個(gè)context。
雖然放多個(gè)context的動(dòng)態(tài)構(gòu)造還有待研究,with 后面的代碼塊也不能填一個(gè)元組或者列表。。惆悵。。
好啦!今天的分享到這里就結(jié)束了,希望大家持續(xù)關(guān)注馬哥教育官網(wǎng),每天都會(huì)有大量?jī)?yōu)質(zhì)內(nèi)容與大家分享!聲明:文章轉(zhuǎn)載于網(wǎng)絡(luò),版權(quán)歸原作者所有,如有侵權(quán),請(qǐng)及時(shí)聯(lián)系刪除!