轉載:簡單介紹Python中的try和finally和with方法

来源:http://www.cnblogs.com/alan-babyblog/archive/2016/01/23/5153343.html
-Advertisement-
Play Games

用 Python 做一件很平常的事情: 打開文件, 逐行讀入, 最後關掉文件; 進一步的需求是, 這也許是程式中一個可選的功能, 如果有任何問題, 比如文件無法打開, 或是讀取出錯, 那麼在函數內需要捕獲所有異常, 輸出一行警告並退出. 代碼可能一開始看起來是這樣的?12345678def read...


用 Python 做一件很平常的事情: 打開文件, 逐行讀入, 最後關掉文件; 進一步的需求是, 這也許是程式中一個可選的功能, 如果有任何問題, 比如文件無法打開, 或是讀取出錯, 那麼在函數內需要捕獲所有異常, 輸出一行警告並退出. 代碼可能一開始看起來是這樣的
 

?
1 2 3 4 5 6 7 8 def read_file():   try:     f = open('yui', 'r')     print ''.join(f.readlines())   except:     print 'error occurs while reading file'   finally:     f.close()

    不過這顯然無法運作, 因為  f  是在  try  塊中定義的, 而在  finally  中無法引用.

    如果將  f  提取到  try  塊外部, 如
 

?
1 2 3 4 5 6 7 8 def read_file():    f = open('azusa', 'r')   try:     print ''.join(f.readlines())   except:     print 'error occurs while reading file'   finally:     f.close()

那麼, 問題在於當打開文件失敗, 拋出異常將不會被捕獲.

    挫一點的方法自然是, 再套一層  try  吧
 

?
1 2 3 4 5 6 7 8 9 10 11 def read_file():    try:     f = open('sawako', 'r')     try:       print ''.join(f.readlines())     except:       print 'error occurs while reading file'     finally:       f.close()    except:      print 'error occurs while reading file'

    當然這不僅僅是多一層縮進挫了, 連警告輸出都白白多一次呢.

    正規一點的方式是, 使用 Python 引入的  with  結構來解決, 如
 

?
1 2 3 4 5 6 def readFile():   try:      with open('mio', 'r') as f:       print ''.join(f.readlines())   except:     print 'error occurs while reading file'

    當文件打開失敗時, 異常自然會被  except  到; 否則, 在  with  塊結束之後, 打開的文件將自動關閉.

    除了打開文件, 還有其它這樣可以用於  with  的東西麽? 或者說, 怎麼自定義一個什麼東西, 讓它能用於  with 呢?
    直接回答後一個問題吧, 秘密在於 Python 虛擬機在  with  塊退出時會去尋找對象的  __exit__  方法並調用它, 把釋放資源的動作放在這個  __exit__  函數中就可以了; 另外, 對象還需要一個  __enter__  函數, 當進入  with 塊時, 這個函數被調用, 而它的返回值將作為  as  後引用的值. 一個簡單的例子是
 

?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 class Test:   def __init__(self):     print 'init'      def __enter__(self):     print 'enter'     return self      def __exit__(self, except_type, except_obj, tb):     print except_type     print except_obj     import traceback     print ''.join(traceback.format_tb(tb))     print 'exit'     return True    with Test() as t:   raise ValueError('kon!')

    執行這一段代碼, 輸出將會是
 

?
1 2 3 4 5 6 7 8 init enter <type 'exceptions.ValueError'> kon!  File "test.py", line 17, in <module>   raise ValueError('kon!')    exit

     __exit__  函數接受三個參數, 分別是異常對象類型, 異常對象和調用棧. 如果  with  塊正常退出, 那麼這些參數將都是  None . 返回  True  表示發生的異常已被處理, 不再繼續向外拋出.

    簡單的介紹到此為止, 詳細的情況可以參考  PEP 343  (這數字真不錯, 7 3 ).

下麵介紹下 with 語句的實例用法 & 高級用法:

Python高端、大氣、上檔次的with語句

在說with語句之前,先看看一段簡單的代碼吧
 

?
1 2 3 4 5 lock = threading.Lock() ... lock.acquire() elem = heapq.heappop(heap) lock.release()

很簡單直觀,多個線程共用一個優先順序隊列的時候,首先先用互斥鎖lock.acquire()把優先順序隊列鎖上,然後取元素,再然後lock.release()釋放這個鎖。

雖然看似非常符合邏輯的一個過程,但是裡面隱藏著一個巨大的bug:當heap裡面沒有元素的時候,會拋出一個IndexError異常,再然後堆棧回滾,再然後lock.release()根本不會執行,這個鎖就永遠得不到釋放,因此就發生了喜聞樂見的死鎖問題。這個也是很多大神們討厭異常的原因。經典Java風格的解決方案就是
 

?
1 2 3 4 5 6 7 lock = threading.Lock() ... lock.acquire() try:   elem = heapq.heappop(heap) finally:   lock.release()

這個雖然可以,但是怎麼看怎麼dirty,和Python優雅、簡單的風格出入很大。其實,自從Python2.5開始引入了with語句,一切就變得非常簡單:
 

?
1 2 3 4 lock = threading.Lock() ... with lock:   elem = heapq.heappop(heap)

在此無論以何種方式離開with語句的代碼塊,鎖都會被釋放。
with語句的設計目的就是為了使得之前需要通過try...finally解決的清理資源問題變得簡單、清晰,它的的用法是
 

?
1 2 with expression [as variable]:   with-block

其中expression返回一個叫做「context manager」的對象,然後這個對象被賦給variable(如果有的話)。「context manager」對象有兩個方法,分別是__enter__()和__exit__(),很明顯一個在進入with-block時調用,一個離開with-block的時候調用。

這樣的對象不需要自己去實現,在Python標準庫裡面很多API都是已經實現了這兩個方法,最常見的一個例子就是讀寫文件的open語句。
 

?
1 2 with open('1.txt', encoding = 'utf-8') as fp:   lines = fp.readlines()

無論是正常離開還是因為異常原因離開with語句塊,打開的文件資源總是會釋放。
接下去討論一下with語句配合contextlib庫的一些比較實用的方法,比如需要同時打開兩個文件,一個讀一個寫,這個時候就可以這樣寫:
 

?
1 2 3 4 from contextlib import nested ... with nested(open('in.txt'), open('out.txt', 'w')) as (fp_in, fp_out):   ...

這樣就可以省掉兩個with的語句的嵌套了,另外如果遇到一些還沒有支持「context manager」的API呢?比如urllib.request.urlopen(),這個返回的對象因為不是「context manager」,結束的時候還需要自己去調用close方法。
類似這種API,contextlib提供了一個叫做closing方法,它會在離開with語句的時候,自動調用對象的close方法,因此urlopen也可以這樣寫:
 

?
1 2 3 4 5 from contextlib import closing ... with closing(urllib.request.urlopen('http://www.yahoo.com')) as f:   for line in f:     sys.stdout.write(line)

 用 Python 做一件很平常的事情: 打開文件, 逐行讀入, 最後關掉文件; 進一步的需求是, 這也許是程式中一個可選的功能, 如果有任何問題, 比如文件無法打開, 或是讀取出錯, 那麼在函數內需要捕獲所有異常, 輸出一行警告並退出. 代碼可能一開始看起來是這樣的
 

?
1 2 3 4 5 6 7 8 def read_file():   try:     f = open('yui', 'r')     print ''.join(f.readlines())   except:     print 'error occurs while reading file'   finally:     f.close()

    不過這顯然無法運作, 因為  f  是在  try  塊中定義的, 而在  finally  中無法引用.

    如果將  f  提取到  try  塊外部, 如
 

?
1 2 3 4 5 6 7 8 def read_file():    f = open('azusa', 'r')   try:     print ''.join(f.readlines())   except:     print 'error occurs while reading file'   finally:     f.close()

那麼, 問題在於當打開文件失敗, 拋出異常將不會被捕獲.

    挫一點的方法自然是, 再套一層  try  吧
 

?
1 2 3 4 5 6 7 8 9 10 11 def read_file():    try:     f = open('sawako', 'r')     try:       print ''.join(f.readlines())     except:       print 'error occurs while reading file'     finally:       f.close()    except:      print 'error occurs while reading file'

    當然這不僅僅是多一層縮進挫了, 連警告輸出都白白多一次呢.

    正規一點的方式是, 使用 Python 引入的  with  結構來解決, 如
 

?
1 2 3 4 5 6 def readFile():   try:      with open('mio', 'r') as f:       print ''.join(f.readlines())   except:     print 'error occurs while reading file'

    當文件打開失敗時, 異常自然會被  except  到; 否則, 在  with  塊結束之後, 打開的文件將自動關閉.

    除了打開文件, 還有其它這樣可以用於  with  的東西麽? 或者說, 怎麼自定義一個什麼東西, 讓它能用於  with 呢?
    直接回答後一個問題吧, 秘密在於 Python 虛擬機在  with  塊退出時會去尋找對象的  __exit__  方法並調用它, 把釋放資源的動作放在這個  __exit__  函數中就可以了; 另外, 對象還需要一個  __enter__  函數, 當進入  with 塊時, 這個函數被調用, 而它的返回值將作為  as  後引用的值. 一個簡單的例子是
 

?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 class Test:   def __init__(self):     print 'init'      def __enter__(self):     print 'enter'     return self      def __exit__(self, except_type, except_obj, tb):     print except_type     print except_obj     import traceback     print ''.join(traceback.format_tb(tb))     print 'exit'     return True    with Test() as t:   raise ValueError('kon!')

    執行這一段代碼, 輸出將會是
 

?
1 2 3 4 5 6 7 8 init enter <type 'exceptions.ValueError'> kon!  File "test.py", line 17, in <module>   raise ValueError('kon!')    exit

     __exit__  函數接受三個參數, 分別是異常對象類型, 異常對象和調用棧. 如果  with  塊正常退出, 那麼這些參數將都是  None . 返回  True  表示發生的異常已被處理, 不再繼續向外拋出.

    簡單的介紹到此為止, 詳細的情況可以參考  PEP 343  (這數字真不錯, 7 3 ).

下麵介紹下 with 語句的實例用法 & 高級用法:

Python高端、大氣、上檔次的with語句

在說with語句之前,先看看一段簡單的代碼吧
 

?
1 2 3 4 5 lock = threading.Lock() ... lock.acquire() elem = heapq.heappop(heap) lock.release()

很簡單直觀,多個線程共用一個優先順序隊列的時候,首先先用互斥鎖lock.acquire()把優先順序隊列鎖上,然後取元素,再然後lock.release()釋放這個鎖。

雖然看似非常符合邏輯的一個過程,但是裡面隱藏著一個巨大的bug:當heap裡面沒有元素的時候,會拋出一個IndexError異常,再然後堆棧回滾,再然後lock.release()根本不會執行,這個鎖就永遠得不到釋放,因此就發生了喜聞樂見的死鎖問題。這個也是很多大神們討厭異常的原因。經典Java風格的解決方案就是
 

?
1 2 3 4 5 6 7 lock = threading.Lock() ... lock.acquire() try:   elem = heapq.heappop(heap) finally:   lock.release()

這個雖然可以,但是怎麼看怎麼dirty,和Python優雅、簡單的風格出入很大。其實,自從Python2.5開始引入了with語句,一切就變得非常簡單:
 

?
1 2 3 4 lock = threading.Lock() ... with lock:   elem = heapq.heappop(heap)

在此無論以何種方式離開with語句的代碼塊,鎖都會被釋放。
with語句的設計目的就是為了使得之前需要通過try...finally解決的清理資源問題變得簡單、清晰,它的的用法是
 

?
1 2 with expression [as variable]:   with-block

其中expression返回一個叫做「context manager」的對象,然後這個對象被賦給variable(如果有的話)。「context manager」對象有兩個方法,分別是__enter__()和__exit__(),很明顯一個在進入with-block時調用,一個離開with-block的時候調用。

這樣的對象不需要自己去實現,在Python標準庫裡面很多API都是已經實現了這兩個方法,最常見的一個例子就是讀寫文件的open語句。
 

?
1 2 with open('1.txt', encoding = 'utf-8') as fp:   lines = fp.readlines()

無論是正常離開還是因為異常原因離開with語句塊,打開的文件資源總是會釋放。
接下去討論一下with語句配合contextlib庫的一些比較實用的方法,比如需要同時打開兩個文件,一個讀一個寫,這個時候就可以這樣寫:
 

?
1 2 3 4 from contextlib import nested ... with nested(open('in.txt'), open('out.txt', 'w')) as (fp_in, fp_out):   ...

這樣就可以省掉兩個with的語句的嵌套了,另外如果遇到一些還沒有支持「context manager」的API呢?比如urllib.request.urlopen(),這個返回的對象因為不是「context manager」,結束的時候還需要自己去調用close方法。
類似這種API,contextlib提供了一個叫做closing方法,它會在離開with語句的時候,自動調用對象的close方法,因此urlopen也可以這樣寫:
 

?
1 2 3 4 5 from contextlib import closing ... with closing(urllib.request.urlopen('http://www.yahoo.com')) as f:   for line in f:     sys.stdout.write(line)

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1、理解控制器mvc控制器負責響應來自MVC網站的界面請求,每一個瀏覽器請求都映射到了一個控制器,設想在瀏覽器裡面輸入了http://www.test.com/query/index/3在這種情況下,將會調用一個名為queryController的控制器。queryController負責生成對瀏覽...
  • 之所以為中集,是因為上集遺留的問題並沒有完全解決,但已經有很大的進步,我覺得對於我來說,現階段是可以接受的,簡單總結下上集遺留的三個問題或計劃:1. 嘗試解決 ASP.NET Core 1.0 中解密 Forms Authentication 生成的 Cookie。2. 嘗試解決 ASP.NET C...
  • 在ASP.NET Web API中實現緩存大致有2種思路。一種是通過ETag, 一種是通過類似ASP.NET MVC中的OutputCache。通過ETag實現緩存首先安裝cachecow.serverinstall-package cachecow.server在WebApiConfig中。pub...
  • 一、目前在ASP.NET中頁面傳值共有這麼幾種方式:1、表單提交, .... form1.submit(); .... 此種方在ASP。NET中無效,因為ASP。NET的表單總是提交到自身頁面,如果要提交到別一頁面,需要特殊處理。2、鏈接地址傳送接收頁面: string str = Reques.....
  • 有時候我們可能不想顯示某個實體中的所有欄位。比如客戶端發出如下請求:locaohost:43321/api/groups/1/items?fields=idlocaohost:43321/api/groups/1/items?fields=id,name以上,對於Item這個類,我們可能只想顯示id...
  • 概念:演算法文章,總是帶給我們無窮的思考和興趣,一個問題,多種解決方法,看你如何去思考它,對於標題所引出的問題,我覺得,使用遞歸是比較有效的方法,當然遞歸還有很多使用場合,如樹型分類列表的操作等等。註意:使用遞歸時,初學者要特別註意的就是“出口”,必須為遞歸提供一個出口,否則你的記憶體就要溢出了,呵呵,...
  • Python’s with statement provides a very convenient way of dealing with the situation where you have to do a setup and teardown to make something happe...
  • 我們大家有知道PHP串列化可以把變數包括對象,轉化成連續bytes數據,你可以將串列化後的變數存在一個文件里或在網路上傳輸,然後再反串列化還原為原來的數據。文章這裡就PHP串列化為大家詳細的介紹。你在反串列化類的對象之前定義的類,PHP可以成功地存儲其對象的屬性和方法. 有時你可能需要一個對象在反串...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...