上下文管理协议由__enter__和__exit__方法构成,支持with语句自动管理资源:进入时调用__enter__(返回值赋给as变量),退出时必调__exit__(可抑制异常)。

Python 中的上下文管理协议(Context Management Protocol)通过 __enter__ 和 __exit__ 两个特殊方法实现,让对象能用于 with 语句中,自动完成资源获取与释放,避免手动调用清理逻辑。
什么是上下文管理协议
只要一个类实现了 __enter__ 和 __exit__ 方法,它就支持上下文管理协议。执行 with obj as x: 时:
-
__enter__在进入with块前被调用,返回值赋给as后的变量(若无as,返回值被忽略); -
__exit__(exc_type, exc_value, traceback)在离开with块时被调用,无论是否发生异常都会执行,用于清理工作; - 若
__exit__返回True,会抑制异常(不向上抛出);返回None或False则正常传播异常。
写一个简单的自定义 with 对象
比如封装一个计时器,在进入时记录开始时间,退出时打印耗时:
import time
<p>class Timer:
def <strong>init</strong>(self, name="block"):
self.name = name
self.start_time = None
self.end_time = None</p><pre class="brush:php;toolbar:false;">def __enter__(self):
self.start_time = time.time()
return self # 可选:返回自身或任意值供 as 使用
def __exit__(self, exc_type, exc_value, traceback):
self.end_time = time.time()
elapsed = self.end_time - self.start_time
print(f"[{self.name}] 执行耗时: {elapsed:.4f} 秒")
# 不返回 True,不抑制异常
立即学习“Python免费学习笔记(深入)”;
使用方式:
with Timer("数据处理"):
time.sleep(1.2)
输出类似:[数据处理] 执行耗时: 1.2005 秒
本文档主要讲述的是SCA介绍及应用实例;SCA(Service Component Architecture)是针对SOA提出的一套服务体系构建框架协议,内部既融合了IOC的思想,同时又把面向对象的复用由代码复用上升到了业务模块组件复用,同时将服务接口,实现,部署,调用完全分离,通过配置的形式灵活的组装,绑定。希望本文档会给有需要的朋友带来帮助;感兴趣的朋友可以过来看看
带异常处理的上下文管理器
如果希望在发生特定异常时静默处理(如忽略 FileNotFoundError),可在 __exit__ 中判断并返回 True:
class SafeFileOpener:
def __init__(self, filename, mode="r"):
self.filename = filename
self.mode = mode
self.file = None
<pre class="brush:php;toolbar:false;">def __enter__(self):
self.file = open(self.filename, self.mode)
return self.file
def __exit__(self, exc_type, exc_value, traceback):
if self.file:
self.file.close()
if exc_type is FileNotFoundError:
print(f"警告:文件 {self.filename} 不存在,已跳过")
return True # 抑制 FileNotFoundError
return False # 其他异常照常抛出
立即学习“Python免费学习笔记(深入)”;
这样写就不会因文件不存在而中断程序:
with SafeFileOpener("missing.txt") as f:
content = f.read() # 若文件不存在,只打印警告,不报错
更简洁的方式:用 contextlib.contextmanager
对于简单逻辑,无需写完整类,可用装饰器 @contextmanager 将生成器函数转为上下文管理器:
from contextlib import contextmanager
<p>@contextmanager
def timer(name="block"):
start = time.time()
try:
yield # 进入 with 块的位置
finally:
end = time.time()
print(f"[{name}] 耗时: {end - start:.4f} 秒")</p><h1>使用方式相同</h1><p>with timer("循环计算"):
sum(i * i for i in range(10**6))
</p>yield 之前是 __enter__,之后(含 finally)是 __exit__ 的等价逻辑,更轻量易读。









