一、简介
with是从Python 2.5 引入的一个新的语法,更准确的说,是一种上下文的管理协议,用于简化try…except…finally的处理流程。with通过__enter__方法初始化,然后在__exit__中做善后以及处理异常。对于一些需要预先设置,事后要清理的一些任务,with提供了一种非常方便的表达。
with的基本语法如下,EXPR是一个任意表达式,VAR是一个单一的变量(可以是tuple),”as VAR”是可选的。
二、实现方式
根据前面对with的翻译可以看到,被with求值的对象必须有一个__enter__方法和一个__exit__方法。稍微看一个文件读取的例子吧,注意在这里我们要解决2个问题:文件读取异常,读取完毕后关闭文件句柄。用try…except一般会这样写:
def __init__(self, name):
self.handle = open(name)
def __enter__(self):
return self.handle
def __exit__(self, type, value, trackback):
self.handle.close()
with opened('/tmp/a.txt') as f:
for line in f.readlines():
print(line)
如果你不喜欢定义class,还可以用Python标准库提供的contextlib来实现:
@contextmanager
def opened(name):
f = open(name)
try:
yield f
finally:
f.close()
with opened('/tmp/a.txt') as f:
for line in f.readlines():
print(line)
3. アプリケーションのシナリオ
あまりにもナンセンスなので、どのようなシナリオで使用すべきでしょうか?もちろん、そうでなければこの記事の意味はありません。以下は PEP 343 からの抜粋です。
実行前にコードがロックされ、実行後にロックが解放されることを保証するテンプレート:
with locked(myLock):
# ここのコードは myLock が保持された状態で実行されます。ロックは
# ブロックが離れると解放されることが保証されます (
# リターン経由またはキャッチされなかった例外の場合) ).
with opens(filename, "w") as f:
with stdout_redirected(f):
print "Hello world"
4. 概要
With は、try...expect...finally 構文を簡略化したもので、例外を処理するための非常に優れた方法を提供します。 Python の構文を使用して実装するには、クラスベースとデコレータベースの 2 つの方法があり、これら 2 つの方法は原則として同等であり、特定のシナリオに応じて選択できます。
with はもともとブロック...as... 構文から生まれましたが、この構文は多くの人によって反対され、最終的に with が誕生しました。この歴史については、現在でも PEP-343 と PEP-340 を参照してください。 🎜>