日韩久久久精品,亚洲精品久久久久久久久久久,亚洲欧美一区二区三区国产精品 ,一区二区福利

Python 中的上下文管理器

系統(tǒng) 1787 0

Python 中的上下文管理器

            
              
                with
              
               expression 
              
                [
              
              
                as
              
               target
              
                ]
              
              
                :
              
              
                with
              
              
                -
              
              body

            
          

上下文管理器是為with 語(yǔ)句而生。只要實(shí)現(xiàn)了上下文管理器協(xié)議 __enter__ __exit__ ,就可以使用with語(yǔ)句。

__enter__ 通常執(zhí)行一些初始化操作,并且該函數(shù)的返回值會(huì)賦值給可選的 as target 中的 target 變量。

__exit__ 執(zhí)行資源清理工作。它接收三個(gè)參數(shù),異常類型,異常實(shí)例,和異常棧,根據(jù)這些異常信息, __exit__ 可以選擇進(jìn)行相應(yīng)的異常處理,并默認(rèn)拋出異常。如果我們?cè)谧? __exit__ 返回True,相當(dāng)于告訴python:這些異常我都已經(jīng)處理了,都在掌控之中,您老不必操心。

除了自定義類手動(dòng)實(shí)現(xiàn)兩個(gè)特殊方法外,還有另一種途徑實(shí)現(xiàn)一個(gè)上下文管理器。
標(biāo)準(zhǔn)庫(kù) contextlib 中提供了一個(gè) @contextmanager 可以方便的把一個(gè)協(xié)程函數(shù)包裝成一個(gè)上下文管理器。
《Fluent Python》 書中一個(gè)好玩的例子:

            
              @contextmanager

              
                def
              
              
                f
              
              
                (
              
              
                )
              
              
                :
              
              
                import
              
               sys
    
              
                print
              
              
                (
              
              
                '歡迎來(lái)到鏡像的世界'
              
              
                )
              
              
    origin_print 
              
                =
              
               sys
              
                .
              
              stdout
              
                .
              
              write
    sys
              
                .
              
              stdout
              
                .
              
              write 
              
                =
              
              
                lambda
              
               x
              
                :
              
               origin_print
              
                (
              
              x
              
                [
              
              
                :
              
              
                :
              
              
                -
              
              
                1
              
              
                ]
              
              
                )
              
              
                # 初始化:替換系統(tǒng)輸入。運(yùn)行中動(dòng)態(tài)修改、添加類的方法————猴子補(bǔ)丁。
              
              
                yield
              
              
                '這里的打印都是反向輸出'
              
              
    sys
              
                .
              
              stdout
              
                .
              
              write 
              
                =
              
               origin_print  
              
                # 退出時(shí):恢復(fù)系統(tǒng)輸入
              
              
                print
              
              
                (
              
              
                'Finally I come back'
              
              
                )
              
              

mirror_world 
              
                =
              
               f
              
                (
              
              
                )
              
              
                with
              
               mirror_world 
              
                as
              
               target
              
                :
              
              
                print
              
              
                (
              
              target
              
                )
              
            
          

輸出結(jié)果:

            
              歡迎來(lái)到鏡像的世界
出輸向反是都印打的里這
Finally I come back

            
          

協(xié)程函數(shù)中yield之前的所有代碼相當(dāng)于 __enter__ 部分的工作,執(zhí)行初始化,執(zhí)行中動(dòng)態(tài)替換了系統(tǒng)的輸出功能(猴子補(bǔ)丁特性)。
并且把一個(gè)結(jié)果綁定到 with...as target target 。至此協(xié)程函數(shù)交出代碼執(zhí)行權(quán),python轉(zhuǎn)而去執(zhí)行with-block里面的代碼。執(zhí)行完with-block 開始執(zhí)行yield之后的代碼——相當(dāng)于 __exit__ 的工作,執(zhí)行資源清理。

至此我們好像實(shí)現(xiàn)了一個(gè)功能正常的上下文管理器。但別忘了還有異常捕獲的機(jī)制。。。

在終端中執(zhí)行 mirror_world 時(shí),如果with-block中拋出了一個(gè)異常,會(huì)導(dǎo)致資源清理工作沒(méi)有進(jìn)行,之后所有的print仍是反向輸出。我們還應(yīng)做的是把yield行的代碼包裹在一個(gè) try...except...finally 中,在finally-bolck中執(zhí)行資源清理工作,以保證正常退出(鬼知道用戶會(huì)在with-block搞什么蛇皮…)。


更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號(hào)聯(lián)系: 360901061

您的支持是博主寫作最大的動(dòng)力,如果您喜歡我的文章,感覺(jué)我的文章對(duì)您有幫助,請(qǐng)用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長(zhǎng)非常感激您!手機(jī)微信長(zhǎng)按不能支付解決辦法:請(qǐng)將微信支付二維碼保存到相冊(cè),切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對(duì)您有幫助就好】

您的支持是博主寫作最大的動(dòng)力,如果您喜歡我的文章,感覺(jué)我的文章對(duì)您有幫助,請(qǐng)用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長(zhǎng)會(huì)非常 感謝您的哦!!!

發(fā)表我的評(píng)論
最新評(píng)論 總共0條評(píng)論
主站蜘蛛池模板: 南通市| 山丹县| 沐川县| 广平县| 东辽县| 乌拉特前旗| 白山市| 汕尾市| 苍溪县| 东丰县| 太保市| 子洲县| 文安县| 曲麻莱县| 若羌县| 汤原县| 临邑县| 宝坻区| 平山县| 沐川县| 安平县| 奎屯市| 平潭县| 扶风县| 茂名市| 庆安县| 扶沟县| 武定县| 鹤山市| 大连市| 青岛市| 青铜峡市| 深泽县| 时尚| 卓资县| 天长市| 鹤岗市| 印江| 甘泉县| 县级市| 报价|