1 模块简介
在数年前,Python 2.5 加入了一个非常特殊的关键字,就是with。with语句允许开发者创建上下文管理器。什么是上下文管理器?上下文管理器就是允许你可以自动地开始和结束一些事情。例如,你可能想要打开一个文件,然后写入一些内容,最后再关闭文件。这或许就是上下文管理器中一个最经典的示例。事实上,当你利用with语句打开一个文件时,Python替你自动创建了一个上下文管理器。
with open("test/test.txt","w") as f_obj:
f_obj.write("hello")
如果你使用的是Python 2.4,你不得不以一种老的方式来完成这个任务
f_obj = open("test/test.txt","w")
f_obj.write("hello")
f_obj.close()
下文管理器背后工作的机制是使用Python的方法:__enter__和__exit__。让我们尝试着去创建我们的上下文管理器,以此来了解上下文管理器是如何工作的。
2 模块使用
2.0 一个简单的demo
import contextlib
import time @contextlib.contextmanager
def timeit(title):
print('1...')
start = time.time()
yield
print('2...')
end = time.time()
usedTime = (end - start) * 1000
print('Use time %d ms' % usedTime) with timeit(1):
print('3...')
time.sleep(1) with timeit(2):
print('4...')
time.sleep(2)
输出结果:
1...
3...
2...
Use time 1001 ms
1...
4...
2...
Use time 2002 ms
2.1 创建一个上下文管理器类
与其继续使用Python打开文件这个例子,不如我们创建一个上下文管理器,这个上下文管理器将会创建一个SQLite数据库连接,当任务处理完毕,将会将其关闭。下面就是一个简单的示例。
import sqlite3 class DataConn:
def __init__(self,db_name):
self.db_name = db_name def __enter__(self):
self.conn = sqlite3.connect(self.db_name)
return self.conn def __exit__(self,exc_type,exc_val,exc_tb):
self.conn.close()
if exc_val:
raise if __name__ == "__main__":
# with 装饰器的底层原理实际上是__enter__和__exit__ 实现的。
db = "test/test.db"
with DataConn(db) as conn:
cursor = conn.cursor()
2.2 利用contextlib创建一个上下文管理器
Python 2.5 不仅仅添加了with语句,它也添加了contextlib模块。这就允许我们使用contextlib的contextmanager函数作为装饰器,来创建一个上下文管理器。让我们尝试着用它来创建一个上下文管理器,用于打开和关闭文件
from contextlib import contextmanager @contextmanager
def file_open(path):
try:
f_obj = open(path,"w")
yield f_obj
except OSError:
print("We had an error!")
finally:
print("Closing file")
f_obj.close() if __name__ == "__main__":
with file_open("test/test.txt") as fobj:
fobj.write("Testing context managers")