超直觀精闢python with as的用法及原理

2022-03-01 15:22:30 字數 3118 閱讀 8992

with...as作用和try...finally一樣,也是捕獲異常,但是不處理異常,不管使用過程中是否發生異常都會執行必要的「清理」操作,釋放資源,比如檔案使用後自動關閉等操作with語句是什麼?

有一些任務,可能事先需要設定,事後做清理工作。對於這種場景,python的with語句提供了一種非常方便的處理方式。乙個很好的例子是檔案處理,你需要獲取乙個檔案控制代碼,從檔案中讀取資料,然後關閉檔案控制代碼。

如果不用with語句,**如下:

file = open("/tmp/foo.txt")

data = file.read()

file.close()

這裡有兩個問題。一是可能忘記關閉檔案控制代碼;二是檔案讀取資料發生異常,沒有進行任何處理。下面是處理異常的加強版本:

file = open("/tmp/foo.txt")

try:

data = file.read()

finally:

file.close()

雖然這段**執行良好,但是太冗長了。這時候就是with一展身手的時候了。除了有更優雅的語法,with還可以很好的處理上下文環境產生的異常。下面是with版本的**:

with open("/tmp/foo.txt") as file:

data = file.read()

with如何工作?

這看起來充滿魔法,但不僅僅是魔法,python對with的處理還很聰明。基本思想是with所求值的物件必須有乙個enter()方法,乙個exit()方法。

緊跟with後面的語句被求值後,返回物件的enter()方法被呼叫,這個方法的返回值將被賦值給as後面的變數。當with後面的**塊全部被執行完之後,將呼叫前面返回物件的exit()方法。

下面例子可以具體說明with如何工作:

#!/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

執行**,輸出如下

in __enter__()

sample: foo

in __exit__()

正如你看到的,

1.enter()方法被執行

2.enter()方法返回的值 - 這個例子中是"foo",賦值給變數'sample'

3. 執行**塊,列印變數"sample"的值為 "foo"

4.exit()方法被呼叫

with真正強大之處是它可以處理異常。可能你已經注意到sample類的exit方法有三個引數- val, type 和 trace。 這些引數在異常處理中相當有用。我們來改一下**,看看具體如何工作的。

#!/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()

這個例子中,with後面的get_sample()變成了sample()。這沒有任何關係,只要緊跟with後面的語句所返回的物件有enter()和exit()方法即可。此例中,sample()的enter()方法返回新建立的sample物件,並賦值給變數sample。

**執行後:

bash-3.2$ ./with_example02.py

type: value: integer division or modulo by zero

trace: traceback (most recent call last):

file "./with_example02.py", line 19, in sample.do_something()

file "./with_example02.py", line 15, in do_something

bar = 1/0

zerodivisionerror: integer division or modulo by zero

實際上,在with後面的**塊丟擲任何異常時,exit()方法被執行。正如例子所示,異常丟擲時,與之關聯的type,value和stack trace傳給exit()方法,因此丟擲的zerodivisionerror異常被列印出來了。開發庫時,清理資源,關閉檔案等等操作,都可以放在exit方法當中。

因此,python的with語句是提供乙個有效的機制,讓**更簡練,同時在異常產生時,清理工作更簡單。

你不知道的CSS3選擇器 精闢用法

本文主要講 css3 中三中不常用的選擇器的精闢用法 e f,e f,e not selector 1 e f 選擇e元素後面的所有兄弟元素f 通用兄弟元素選擇器型別。選擇匹配f的所有元素,且匹配元素位於匹配e的元素後面。在dom結構樹中,e和f所匹配的元素應該在同一級結構上。需求 根據後台傳過來的...

react hook超實用的用法和技巧分析

react hook發布也已經有幾個月了,相信有部分人已經開始使用了,還有些人在猶豫要不要用,可能更多人安於現狀,沒有要用的打算,甚至還有很多公司的react版本是15或以下的,迫於公升級的難度沒有使用。以我個人的觀點,要不要使用react hook呢?建議用的的人 專案react版本已經是reac...

list容器用法詳解(超好用的list) c

尊重原創 原創部落格鏈結 吐槽下部落格上 這篇部落格的人不標明出處.找原創找了好久嗚嗚嗚 1.關於list容器list是一種序列式容器。list容器完成的功能實際上和資料結構中的雙向鍊錶是極其相似的,list中的資料元素是通過鍊錶指標串連成邏輯意義上的線性表,也就是list也具有鍊錶的主要優點,即 ...