with是從Python2.5引入的一個新的語法,它是一種上下文管理協議,目的在於從流程圖中把 try,except 和finally 關鍵字和
資源分配釋放相關代碼統統去掉,簡化try….except….finlally的處理流程。
with通過__enter__方法初始化,然后在__exit__中做善后以及處理異常。
所以使用with處理的對象必須有__enter__()和__exit__()這兩個方法。
其中__enter__()方法在語句體(with語句包裹起來的代碼塊)執行之前進入運行,__exit__()方法在語句體執行完畢退出后運行。
with 語句適用於對資源進行訪問的場合,確保不管使用過程中是否發生異常都會執行必要的“清理”操作,釋放資源,比如文件使用后自動關閉、線程中鎖的自動獲取和釋放等。
With語句的基本語法格式:
with expression [as target]:
with_body
參數說明:
expression:是一個需要執行的表達式;
target:是一個變量或者元組,存儲的是expression表達式執行返回的結果,可選參數。
例如:

>>> with open('d:\\xxx.txt') as fp:
... print fp.read()
...
我無法大是大非
我無法大是大非
我無法大是大非
with語句的工作原理:
緊跟with后面的語句會被求值,返回對象的__enter__()方法被調用,這個方法的返回值將被賦值給as關鍵字后面的變量,當with后面的代碼塊全部被執行完之后,將調用前面返回對象的__exit__()方法。
with語句最關鍵的地方在於被求值對象必須有__enter__()和__exit__()這兩個方法,那我們就可以通過自己實現這兩方法來自定義with語句處理異常。
示例代碼:
#encoding=utf-8
class opened(object):
def __init__(self,filename):
self.handle=open(filename)
print "Resource:%s"%filename
def __enter__(self):
print "[enter%s]: Allocate resource."%self.handle
return self.handle#可以返回不同的對象
def __exit__(self,exc_type,exc_value,exc_trackback):
print "[Exit %s]: Free resource." %self.handle
if exc_trackback is None:
print "[Exit %s]:Exited without exception."%self.handle
self.handle.close()
else:
print "[Exit %s]: Exited with exception raised."%self.handle
return False # 可以省略,缺省的None也是被看做是False
with opened(r'd:\\xxx.txt') as fp:
for line in fp.readlines():
print line
結果:

opened中的__enter__() 返回的是自身的引用,這個引用可以賦值給 as 子句中的fp變量;
返回值的類型可以根據實際需要設置為不同的類型,不必是上下文管理器對象本身。
__exit__() 方法中對變量exc_trackback進行檢測,如果不為 None,表示發生了異常,返回 False 表示需要由外部代碼邏輯對異常進行處理;
如果沒有發生異常,缺省的返回值為 None,在布爾環境中也是被看做 False,但是由於沒有異常發生,__exit__() 的三個參數都為 None,上下文管理代碼可以檢測這種情況,做正常處理。__exit__()方法的3個參數,分別代表異常的類型、值、以及堆棧信息。
