with 语句是从 Python 2.5 开始引入的一种与异常处理相关的功能(2.5 版本中要通过 from __future__ import with_statement 导入后才可以使用),从 2.6 版本开始缺省可用(参考 What’s new in Python 2.6?中 with 语句相关部分介绍)。with 语句适用于对资源进行访问的场合,确保不管使用过程中是否发生异常都会执行必要的“清理”操作,释放资源,比如文件使用后自动关闭、线程中锁的自动获取和释放等。
要使用 with 语句,首先要明白上下文管理器这一概念。有了上下文管理器,with 语句才能工作。
下面是一组与上下文管理器和with 语句有关的概念。
上下文管理协议(Context Management Protocol):包含方法 __enter__() 和 __exit__(),支持
该协议的对象要实现这两个方法。
上下文管理器(Context Manager):支持上下文管理协议的对象,这种对象实现了
__enter__() 和 __exit__() 方法。上下文管理器定义执行 with 语句时要建立的运行时上下文,
负责执行 with 语句块上下文中的进入与退出操作。通常使用 with 语句调用上下文管理器,
也可以通过直接调用其方法来使用。
somefile = open(r'somefileName')
try
:
for
line
in
somefile
:
print
line
# ...more code
finally
:
somefile
.
close
(
)
with
open
(
r
'somefileName'
)
as
somefile
:
for
line
in
somefile
:
print
line
# ...more code
这两者的功效是一样的 但是with简单的多。
with:源码
context_manager = context_expression
exit
=
type
(
context_manager
)
.
__exit__
value
=
type
(
context_manager
)
.
__enter__
(
context_manager
)
exc
=
True
# True 表示正常执行,即便有异常也忽略;False 表示重新抛出异常,需要对异常进行处理
try
:
try
:
target
=
value
# 如果使用了 as 子句
with
-
body
# 执行 with-body
except
:
# 执行过程中有异常发生
exc
=
False
# 如果 __exit__ 返回 True,则异常被忽略;如果返回 False,则重新抛出异常
# 由外层代码对异常进行处理
if
not
exit
(
context_manager
,
*
sys
.
exc_info
(
)
)
:
raise
finally
:
# 正常退出,或者通过 statement-body 中的 break/continue/return 语句退出
# 或者忽略异常退出
if
exc
:
exit
(
context_manager
,
None
,
None
,
None
)
# 缺省返回 None,None 在布尔上下文中看做是 False