轉: 理解Python的With語句

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

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...


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 happen. A very good example for this is the situation where you want to gain a handler to a file, read data from the file and the close the file handler. 有一些任務,可能事先需要設置,事後做清理工作。對於這種場景,Python的with語句提供了一種非常方便的處理方式。一個很好的例子是文件處理,你需要獲取一個文件句柄,從文件中讀取數據,然後關閉文件句柄。 Without the with statement, one would write something along the lines of: 如果不用with語句,代碼如下:

1 2 3 file = open("/tmp/foo.txt") data = file.read() file.close()

There are two annoying things here. First, you end up forgetting to close the file handler. The second is how to handle exceptions that may occur once the file handler has been obtained. One could write something like this to get around this: 這裡有兩個問題。一是可能忘記關閉文件句柄;二是文件讀取數據發生異常,沒有進行任何處理。下麵是處理異常的加強版本:

1 2 3 4 5 file = open("/tmp/foo.txt") try:     data = file.read() finally:     file.close()

While this works well, it is unnecessarily verbose. This is where with is useful. The good thing about with apart from the better syntax is that it is very good handling exceptions. The above code would look like this, when using with: 雖然這段代碼運行良好,但是太冗長了。這時候就是with一展身手的時候了。除了有更優雅的語法,with還可以很好的處理上下文環境產生的異常。下麵是with版本的代碼:

1 2 with open("/tmp/foo.txt") as file:     data = file.read()

with如何工作?

while this might look like magic, the way Python handles with is more clever than magic. The basic idea is that the statement after with has to evaluate an object that responds to an __enter__() as well as an __exit__() function. 這看起來充滿魔法,但不僅僅是魔法,Python對with的處理還很聰明。基本思想是with所求值的對象必須有一個__enter__()方法,一個__exit__()方法。 After the statement that follows with is evaluated, the __enter__() function on the resulting object is called. The value returned by this function is assigned to the variable following as. After every statement in the block is evaluated, the __exit__() function is called. 緊跟with後面的語句被求值後,返回對象的__enter__()方法被調用,這個方法的返回值將被賦值給as後面的變數。當with後面的代碼塊全部被執行完之後,將調用前面返回對象的__exit__()方法。 This can be demonstrated with the following example: 下麵例子可以具體說明with如何工作:

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 #!/usr/bin/env python # with_example01.py     class Sample:     def __enter__(self):         print "In __enter__()"         return "Foo"       def __exit__(self, type, value, trace):         print "In __exit__()"     def get_sample():     return Sample()     with get_sample() as sample:     print "sample:", sample

When executed, this will result in: 運行代碼,輸出如下

1 2 3 4 bash-3.2$ ./with_example01.py In __enter__() sample: Foo In __exit__()

As you can see, The __enter__() function is executed The value returned by it - in this case "Foo" is assigned to sample The body of the block is executed, thereby printing the value of sample ie. "Foo" The __exit__() function is called. What makes with really powerful is the fact that it can handle exceptions. You would have noticed that the __exit__() function for Sample takes three arguments - val, type and trace. These are useful in exception handling. Let’s see how this works by modifying the above example. 正如你看到的, 1. __enter__()方法被執行 2. __enter__()方法返回的值 - 這個例子中是"Foo",賦值給變數'sample' 3. 執行代碼塊,列印變數"sample"的值為 "Foo" 4. __exit__()方法被調用 with真正強大之處是它可以處理異常。可能你已經註意到Sample類的__exit__方法有三個參數- val, type 和 trace。 這些參數在異常處理中相當有用。我們來改一下代碼,看看具體如何工作的。

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 #!/usr/bin/env python # with_example02.py     class Sample:     def __enter__(self):         return self       def __exit__(self, type, value, trace):         print "type:", type         print "value:", value         print "trace:", trace       def do_something(self):         bar = 1/0         return bar + 10   with Sample() as sample:     sample.do_something()

Notice how in this example, instead of get_sample(), with takes Sample(). It does not matter, as long as the statement that follows with evaluates to an object that has an __enter__() and __exit__() functions. In this case, Sample()’s __enter__() returns the newly created instance of Sample and that is what gets passed to sample. 這個例子中,with後面的get_sample()變成了Sample()。這沒有任何關係,只要緊跟with後面的語句所返回的對象有__enter__()和__exit__()方法即可。此例中,Sample()的__enter__()方法返回新創建的Sample對象,並賦值給變數sample。 When executed: 代碼執行後:

1 2 3 4 5 6 7 8 9 10 bash-3.2$ ./with_example02.py type: <type 'exceptions.ZeroDivisionError'> value: integer division or modulo by zero trace: <traceback object at 0x1004a8128> Traceback (most recent call last):   File "./with_example02.py", line 19, in <module>     sample.do_something()   File "./with_example02.py", line 15, in do_something     bar = 1/0 ZeroDivisionError: integer division or modulo by zero

Essentially, if there are exceptions being thrown from anywhere inside the block, the __exit__() function for the object is called. As you can see, the type, value and the stack trace associated with the exception thrown is passed to this function. In this case, you can see that there was a ZeroDivisionError exception being thrown. People implementing libraries can write code that clean up resources, close files etc. in their __exit__() functions. 實際上,在with後面的代碼塊拋出任何異常時,__exit__()方法被執行。正如例子所示,異常拋出時,與之關聯的type,value和stack trace傳給__exit__()方法,因此拋出的ZeroDivisionError異常被列印出來了。開發庫時,清理資源,關閉文件等等操作,都可以放在__exit__方法當中。 Thus, Python’s with is a nifty construct that makes code a little less verbose and makes cleaning up during exceptions a bit easier. 因此,Python的with語句是提供一個有效的機制,讓代碼更簡練,同時在異常產生時,清理工作更簡單。 I have put the code examples given here on Github. 示例代碼可以在Github上面找到。 譯註:本文原文見Understanding Python's "With" Statement


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

-Advertisement-
Play Games
更多相關文章
  • 我實現了XAF項目中Excel數據的導入,使用Devexpress 新出的spreadsheet控制項,可能也不新了吧:D好,先看一下效果圖:下圖是Web版本的。下麵是win版,目前只支持Ribbon UI,下個版本支持其他界面的:功能說明:支持從Excel任意版本導入數據,可以使用 打開文件功能選擇...
  • 在寫一個客戶的B/S結構應用程式時,突然發現一個技巧,不知道是否是MS的一個BUG,給相關的有研究的朋友原先考慮寫一個檢查Session的類,Session失效後,必須轉向登陸頁面,可每一個調用該類的頁面,在不同的WEB路徑下,所以轉到登陸頁面的URL都不同,每個頁面都要調用和設置登陸頁面路徑,所以...
  • 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...
  • 概念:演算法文章,總是帶給我們無窮的思考和興趣,一個問題,多種解決方法,看你如何去思考它,對於標題所引出的問題,我覺得,使用遞歸是比較有效的方法,當然遞歸還有很多使用場合,如樹型分類列表的操作等等。註意:使用遞歸時,初學者要特別註意的就是“出口”,必須為遞歸提供一個出口,否則你的記憶體就要溢出了,呵呵,...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...